diff --git a/search/house_detector.cpp b/search/house_detector.cpp index 3000b356d9..8e6c6049f6 100644 --- a/search/house_detector.cpp +++ b/search/house_detector.cpp @@ -7,6 +7,8 @@ #include "indexer/feature_impl.hpp" #include "indexer/search_string_utils.hpp" +#include "platform/platform.hpp" + #include "geometry/angles.hpp" #include "geometry/distance.hpp" @@ -14,32 +16,29 @@ #include "base/logging.hpp" #include "base/stl_iterator.hpp" -#include "std/bind.hpp" -#include "std/cmath.hpp" -#include "std/exception.hpp" -#include "std/numeric.hpp" -#include "std/set.hpp" -#include "std/string.hpp" +#include +#include +#include +#include +#include +#include +#include +#include +#include + #include "std/transform_iterator.hpp" -#ifdef DEBUG -#include "platform/platform.hpp" - -#include "std/fstream.hpp" -#include "std/iostream.hpp" -#endif - +using namespace std; +using namespace std::placeholders; namespace search { - namespace { - #if 0 void Houses2KML(ostream & s, map const & m) { - for (map::const_iterator it = m.begin(); it != m.end(); ++it) + for (auto it = m.begin(); it != m.end(); ++it) { m2::PointD const & pt = it->first.GetPosition(); @@ -62,10 +61,10 @@ void Street2KML(ostream & s, vector const & pts, char const * color) s << "" << endl; s << "" << endl; + for (size_t i = 0; i < pts.size(); ++i) - { s << MercatorBounds::XToLon(pts[i].x) << "," << MercatorBounds::YToLat(pts[i].y) << "," << "0.0" << endl; - } + s << "" << endl; s << "" << endl; @@ -79,7 +78,6 @@ void Streets2KML(ostream & s, MergedStreet const & st, char const * color) class KMLFileGuard { - ofstream m_file; public: KMLFileGuard(string const & name) { @@ -90,16 +88,18 @@ public: m_file << "" << endl; } - ostream & GetStream() { return m_file; } - ~KMLFileGuard() { m_file << "" << endl; } + + ostream & GetStream() { return m_file; } + +private: + ofstream m_file; }; #endif - double const STREET_CONNECTION_LENGTH_M = 100.0; int const HN_NEARBY_DISTANCE = 4; double const STREET_CONNECTION_MAX_ANGLE = math::pi / 2.0; @@ -108,20 +108,508 @@ size_t const HN_COUNT_FOR_ODD_TEST = 16; //int const HN_NEARBY_INDEX_RANGE = 5; double const HN_MAX_CONNECTION_DIST_M = 300.0; - class StreetCreator { - Street * m_street; public: StreetCreator(Street * st) : m_street(st) {} void operator () (m2::PointD const & pt) const { m_street->m_points.push_back(pt); } + +private: + Street * m_street; }; +bool LessStreetDistance(HouseProjection const & p1, HouseProjection const & p2) +{ + return p1.m_streetDistance < p2.m_streetDistance; } +double GetDistanceMeters(m2::PointD const & p1, m2::PointD const & p2) +{ + return MercatorBounds::DistanceOnEarth(p1, p2); +} + +pair GetConnectionAngleAndDistance(bool & isBeg, Street const * s1, Street const * s2) +{ + m2::PointD const & p1 = isBeg ? s1->m_points.front() : s1->m_points.back(); + m2::PointD const & p0 = isBeg ? s1->m_points[1] : s1->m_points[s1->m_points.size()-2]; + + double const d0 = p1.SquareLength(s2->m_points.front()); + double const d2 = p1.SquareLength(s2->m_points.back()); + isBeg = (d0 < d2); + m2::PointD const & p2 = isBeg ? s2->m_points[1] : s2->m_points[s2->m_points.size()-2]; + + return make_pair(ang::GetShortestDistance(ang::AngleTo(p0, p1), ang::AngleTo(p1, p2)), min(d0, d2)); +} + +class HasSecond +{ +public: + HasSecond(set const & streets) : m_streets(streets) {} + template + bool operator()(T const & t) const + { + return m_streets.count(t.second) > 0; + } + +private: + set const & m_streets; +}; + +class HasStreet +{ +public: + HasStreet(set const & streets) : m_streets(streets) {} + bool operator()(MergedStreet const & st) const + { + for (size_t i = 0; i < st.m_cont.size(); ++i) + { + if (m_streets.count(st.m_cont[i]) > 0) + return true; + } + return false; + } + +private: + set const & m_streets; +}; + +struct ScoredHouse +{ + House const * house; + double score; + ScoredHouse(House const * h, double s) : house(h), score(s) {} + ScoredHouse() : house(0), score(numeric_limits::max()) {} +}; + +class ResultAccumulator +{ +public: + ResultAccumulator(string const & houseNumber) : m_number(houseNumber) {} + + string const & GetFullNumber() const { return m_number.GetNumber(); } + bool UseOdd() const { return m_useOdd; } + + bool SetStreet(MergedStreet const & st) + { + Reset(); + + m_useOdd = true; + m_isOdd = m_number.IsOdd(); + return st.GetHousePivot(m_isOdd, m_sign) != 0; + } + + void SetSide(bool sign) + { + Reset(); + + m_useOdd = false; + m_sign = sign; + } + + void Reset() + { + for (size_t i = 0; i < ARRAY_SIZE(m_results); ++i) + m_results[i] = ScoredHouse(); + } + + void ResetNearby() { m_results[3] = ScoredHouse(); } + + bool IsOurSide(HouseProjection const & p) const + { + if (m_sign != p.m_projSign) + return false; + return (!m_useOdd || m_isOdd == p.IsOdd()); + } + + void ProcessCandidate(HouseProjection const & p) + { + if (IsOurSide(p)) + MatchCandidate(p, false); + } + + void MatchCandidate(HouseProjection const & p, bool checkNearby) + { + int ind = p.m_house->GetMatch(m_number); + if (ind == -1) + { + if (checkNearby && p.m_house->GetNearbyMatch(m_number)) + ind = 3; + else + return; + } + + if (IsBetter(ind, p.m_distMeters)) + m_results[ind] = ScoredHouse(p.m_house, p.m_distMeters); + } + + template + void FlushResults(TCont & cont) const + { + size_t const baseScore = 1 << (ARRAY_SIZE(m_results) - 2); + + for (size_t i = 0; i < ARRAY_SIZE(m_results) - 1; ++i) + { + if (m_results[i].house) + { + // Scores are: 4, 2, 1 according to the matching. + size_t const score = baseScore >> i; + + size_t j = 0; + for (; j < cont.size(); ++j) + { + if (cont[j].first == m_results[i].house) + { + cont[j].second += score; + break; + } + } + + if (j == cont.size()) + cont.push_back(make_pair(m_results[i].house, score)); + } + } + } + + House const * GetBestMatchHouse() const { return m_results[0].house; } + + bool HasBestMatch() const { return (m_results[0].house != 0); } + House const * GetNearbyCandidate() const { return (HasBestMatch() ? 0 : m_results[3].house); } + +private: + bool IsBetter(int ind, double dist) const + { + return m_results[ind].house == 0 || m_results[ind].score > dist; + } + + ParsedNumber m_number; + bool m_isOdd, m_sign, m_useOdd; + + ScoredHouse m_results[4]; +}; +/* +void ProcessNearbyHouses(vector const & v, ResultAccumulator & acc) +{ + House const * p = acc.GetNearbyCandidate(); + if (p) + { + // Get additional search interval. + int const pivot = distance(v.begin(), find_if(v.begin(), v.end(), +HouseProjection::EqualHouse(p))); ASSERT(pivot >= 0 && pivot < v.size(), ()); int const start = +max(pivot - HN_NEARBY_INDEX_RANGE, 0); int const end = min(pivot + HN_NEARBY_INDEX_RANGE + 1, +int(v.size())); + + for (int i = start; i < end; ++i) + acc.MatchCandidate(*v[i], false); + } +} +*/ +// void GetClosestHouse(MergedStreet const & st, ResultAccumulator & acc) +//{ +// for (MergedStreet::Index i = st.Begin(); !st.IsEnd(i); st.Inc(i)) +// acc.ProcessCandidate(st.Get(i)); +//} + +// If it's odd or even part of the street pass 2, else pass 1. +void AddToQueue(int houseNumber, int step, queue & q) +{ + for (int i = 1; i <= 2; ++i) + { + int const delta = step * i; + q.push(houseNumber + delta); + if (houseNumber - delta > 0) + q.push(houseNumber - delta); + } +} + +struct HouseChain +{ + vector houses; + set chainHouses; + double score; + int minHouseNumber; + int maxHouseNumber; + + HouseChain() + { + minHouseNumber = -1; + maxHouseNumber = numeric_limits::max(); + } + + HouseChain(HouseProjection const * h) + { + minHouseNumber = maxHouseNumber = h->m_house->GetIntNumber(); + Add(h); + } + + void Add(HouseProjection const * h) + { + if (chainHouses.insert(h->m_house->GetNumber()).second) + { + int num = h->m_house->GetIntNumber(); + if (num < minHouseNumber) + minHouseNumber = num; + if (num > maxHouseNumber) + maxHouseNumber = num; + houses.push_back(h); + } + } + + bool Find(string const & str) { return (chainHouses.find(str) != chainHouses.end()); } + + void CountScore() + { + sort(houses.begin(), houses.end(), HouseProjection::LessDistance()); + size_t const size = houses.size(); + score = 0; + size_t const scoreNumber = 3; + for (size_t i = 0; i < scoreNumber; ++i) + score += i < size ? houses[i]->m_distMeters : houses.back()->m_distMeters; + score /= scoreNumber; + } + + bool IsIntersecting(HouseChain const & chain) const + { + if (minHouseNumber >= chain.maxHouseNumber) + return false; + if (chain.minHouseNumber >= maxHouseNumber) + return false; + return true; + } + + bool operator<(HouseChain const & p) const { return score < p.score; } +}; + +void GetBestHouseFromChains(vector & houseChains, ResultAccumulator & acc) +{ + for (size_t i = 0; i < houseChains.size(); ++i) + houseChains[i].CountScore(); + sort(houseChains.begin(), houseChains.end()); + + ParsedNumber number(acc.GetFullNumber()); + + for (size_t i = 0; i < houseChains.size(); ++i) + { + if (i == 0 || !houseChains[0].IsIntersecting(houseChains[i])) + { + for (size_t j = 0; j < houseChains[i].houses.size(); ++j) + { + if (houseChains[i].houses[j]->m_house->GetMatch(number) != -1) + { + for (size_t k = 0; k < houseChains[i].houses.size(); ++k) + acc.MatchCandidate(*houseChains[i].houses[k], true); + break; + } + } + } + } +} + +struct Competitiors +{ + uint32_t m_candidateIndex; + uint32_t m_chainIndex; + double m_score; + Competitiors(uint32_t candidateIndex, uint32_t chainIndex, double score) + : m_candidateIndex(candidateIndex), m_chainIndex(chainIndex), m_score(score) + { + } + bool operator<(Competitiors const & c) const { return m_score < c.m_score; } +}; + +void ProccessHouses(vector const & st, ResultAccumulator & acc) +{ + vector houseChains; + size_t const count = st.size(); + size_t numberOfStreetHouses = count; + vector used(count, false); + string const & houseNumber = acc.GetFullNumber(); + int const step = acc.UseOdd() ? 2 : 1; + + for (size_t i = 0; i < count; ++i) + { + HouseProjection const * hp = st[i]; + if (st[i]->m_house->GetNumber() == houseNumber) + { + houseChains.push_back(HouseChain(hp)); + used[i] = true; + --numberOfStreetHouses; + } + } + if (houseChains.empty()) + return; + + queue houseNumbersToCheck; + AddToQueue(houseChains[0].houses[0]->m_house->GetIntNumber(), step, houseNumbersToCheck); + while (numberOfStreetHouses > 0) + { + if (!houseNumbersToCheck.empty()) + { + int candidateHouseNumber = houseNumbersToCheck.front(); + houseNumbersToCheck.pop(); + vector candidates; + ASSERT_LESS(used.size(), numeric_limits::max(), ()); + uint32_t const count = static_cast(used.size()); + for (uint32_t i = 0; i < count; ++i) + { + if (!used[i] && st[i]->m_house->GetIntNumber() == candidateHouseNumber) + candidates.push_back(i); + } + + bool shouldAddHouseToQueue = false; + vector comp; + + for (size_t i = 0; i < candidates.size(); ++i) + { + string num = st[candidates[i]]->m_house->GetNumber(); + ASSERT_LESS(houseChains.size(), numeric_limits::max(), ()); + for (size_t j = 0; j < houseChains.size(); ++j) + { + if (!houseChains[j].Find(num)) + { + double dist = numeric_limits::max(); + for (size_t k = 0; k < houseChains[j].houses.size(); ++k) + { + if (abs(houseChains[j].houses[k]->m_house->GetIntNumber() - + st[candidates[i]]->m_house->GetIntNumber()) <= HN_NEARBY_DISTANCE) + dist = min(dist, GetDistanceMeters(houseChains[j].houses[k]->m_house->GetPosition(), + st[candidates[i]]->m_house->GetPosition())); + } + if (dist < HN_MAX_CONNECTION_DIST_M) + comp.push_back(Competitiors(candidates[i], static_cast(j), dist)); + } + } + } + sort(comp.begin(), comp.end()); + + for (size_t i = 0; i < comp.size(); ++i) + { + if (!used[comp[i].m_candidateIndex]) + { + string num = st[comp[i].m_candidateIndex]->m_house->GetNumber(); + if (!houseChains[comp[i].m_chainIndex].Find(num)) + { + used[comp[i].m_candidateIndex] = true; + houseChains[comp[i].m_chainIndex].Add(st[comp[i].m_candidateIndex]); + --numberOfStreetHouses; + shouldAddHouseToQueue = true; + } + } + } + if (shouldAddHouseToQueue) + AddToQueue(candidateHouseNumber, step, houseNumbersToCheck); + } + else + { + for (size_t i = 0; i < used.size(); ++i) + { + if (!used[i]) + { + houseChains.push_back(HouseChain(st[i])); + --numberOfStreetHouses; + used[i] = true; + AddToQueue(st[i]->m_house->GetIntNumber(), step, houseNumbersToCheck); + break; + } + } + } + } + GetBestHouseFromChains(houseChains, acc); +} + +void GetBestHouseWithNumber(MergedStreet const & st, double offsetMeters, ResultAccumulator & acc) +{ + vector v; + for (MergedStreet::Index i = st.Begin(); !st.IsEnd(i); st.Inc(i)) + { + HouseProjection const & p = st.Get(i); + if (p.m_distMeters <= offsetMeters && acc.IsOurSide(p)) + v.push_back(&p); + } + + ProccessHouses(v, acc); +} + +struct CompareHouseNumber +{ + inline bool Less(HouseProjection const * h1, HouseProjection const * h2) const + { + return (h1->m_house->GetIntNumber() <= h2->m_house->GetIntNumber()); + } + inline bool Greater(HouseProjection const * h1, HouseProjection const * h2) const + { + return (h1->m_house->GetIntNumber() >= h2->m_house->GetIntNumber()); + } +}; +/* +void LongestSubsequence(vector const & v, + vector & res) +{ + LongestSubsequence(v, back_inserter(res), CompareHouseNumber()); +} +*/ +// void GetLSHouse(MergedStreet const & st, double offsetMeters, ResultAccumulator & acc) +//{ +// acc.ResetNearby(); + +// vector v; +// for (MergedStreet::Index i = st.Begin(); !st.IsEnd(i); st.Inc(i)) +// { +// search::HouseProjection const & p = st.Get(i); +// if (p.m_distance <= offsetMeters && acc.IsOurSide(p)) +// v.push_back(&p); +// } + +// vector res; +// LongestSubsequence(v, res); + +// //LOG(LDEBUG, ("=== Offset", offsetMeters, "===")); +// //LogSequence(res); + +// for (size_t i = 0; i < res.size(); ++i) +// acc.MatchCandidate(*(res[i]), true); + +// ProcessNearbyHouses(v, acc); +//} + +struct GreaterSecond +{ + template + bool operator()(T const & t1, T const & t2) const + { + return t1.second > t2.second; + } +}; + +void ProduceVoting(vector const & acc, vector & res, + MergedStreet const & st) +{ + buffer_vector, 4> voting; + + // Calculate score for every house. + for (size_t i = 0; i < acc.size(); ++i) + acc[i].FlushResults(voting); + + if (voting.empty()) + return; + + // Sort according to the score (bigger is better). + sort(voting.begin(), voting.end(), GreaterSecond()); + + // Emit results with equal best score. + size_t const score = voting[0].second; + for (size_t i = 0; i < voting.size(); ++i) + { + if (score == voting[i].second) + res.push_back(HouseResult(voting[i].first, &st)); + else + break; + } +} +} // namespace + ParsedNumber::ParsedNumber(string const & number, bool american) : m_fullN(number) { strings::MakeLowerCaseInplace(m_fullN); @@ -137,40 +625,40 @@ ParsedNumber::ParsedNumber(string const & number, bool american) : m_fullN(numbe { switch (number[curr]) { - case ' ': - case '\t': - ++curr; - break; - case ',': - case ';': - ++curr; - hasComma = true; - break; - case '-': - ++curr; - hasMinus = true; - break; - default: + case ' ': + case '\t': + ++curr; + break; + case ',': + case ';': + ++curr; + hasComma = true; + break; + case '-': + ++curr; + hasMinus = true; + break; + default: + { + if (hasComma || hasMinus) { - if (hasComma || hasMinus) + size_t start = curr; + try { - size_t start = curr; - try - { - int const x = stoi(number.substr(start), &curr, 10); - curr += start; - m_endN = x; - ASSERT_GREATER_OR_EQUAL(m_endN, 0, (number)); - break; - } - catch (exception & e) - { - // Expected case - stoi haven't parsed anything. - } + int const x = stoi(number.substr(start), &curr, 10); + curr += start; + m_endN = x; + ASSERT_GREATER_OR_EQUAL(m_endN, 0, (number)); + break; + } + catch (exception & e) + { + // Expected case - stoi haven't parsed anything. } - curr = 0; - break; } + curr = 0; + break; + } } } @@ -255,44 +743,13 @@ void Street::SetName(string const & name) m_processedName = strings::ToUtf8(GetStreetNameAsKey(name)); } -namespace -{ - -bool LessStreetDistance(HouseProjection const & p1, HouseProjection const & p2) -{ - return p1.m_streetDistance < p2.m_streetDistance; -} - -double GetDistanceMeters(m2::PointD const & p1, m2::PointD const & p2) -{ - return MercatorBounds::DistanceOnEarth(p1, p2); -} - -pair GetConnectionAngleAndDistance(bool & isBeg, Street const * s1, Street const * s2) -{ - m2::PointD const & p1 = isBeg ? s1->m_points.front() : s1->m_points.back(); - m2::PointD const & p0 = isBeg ? s1->m_points[1] : s1->m_points[s1->m_points.size()-2]; - - double const d0 = p1.SquareLength(s2->m_points.front()); - double const d2 = p1.SquareLength(s2->m_points.back()); - isBeg = (d0 < d2); - m2::PointD const & p2 = isBeg ? s2->m_points[1] : s2->m_points[s2->m_points.size()-2]; - - return make_pair(ang::GetShortestDistance(ang::AngleTo(p0, p1), ang::AngleTo(p1, p2)), min(d0, d2)); -} - -} - void Street::Reverse() { ASSERT(m_houses.empty(), ()); reverse(m_points.begin(), m_points.end()); } -void Street::SortHousesProjection() -{ - sort(m_houses.begin(), m_houses.end(), &LessStreetDistance); -} +void Street::SortHousesProjection() { sort(m_houses.begin(), m_houses.end(), &LessStreetDistance); } HouseDetector::HouseDetector(DataSourceBase const & dataSource) : m_loader(dataSource), m_streetNum(0) @@ -390,7 +847,7 @@ void HouseDetector::MergeStreets(Street * st) int HouseDetector::LoadStreets(vector const & ids) { - //LOG(LDEBUG, ("IDs = ", ids)); + // LOG(LDEBUG, ("IDs = ", ids)); ASSERT(IsSortedAndUnique(ids.begin(), ids.end()), ()); @@ -398,19 +855,20 @@ int HouseDetector::LoadStreets(vector const & ids) if (!m_id2st.empty()) { typedef pair ValueT; - function f = bind(&ValueT::first, _1); + function f = bind(&ValueT::first, _1); // Do clear cache if we have elements that are present in the one set, // but not in the other one (set's order is irrelevant). size_t const count = set_intersection(make_transform_iterator(m_id2st.begin(), f), - make_transform_iterator(m_id2st.end(), f), - ids.begin(), ids.end(), - CounterIterator()).GetCount(); + make_transform_iterator(m_id2st.end(), f), ids.begin(), + ids.end(), CounterIterator()) + .GetCount(); if (count < min(ids.size(), m_id2st.size())) { LOG(LDEBUG, ("Clear HouseDetector cache: " - "Common =", count, "Cache =", m_id2st.size(), "Input =", ids.size())); + "Common =", + count, "Cache =", m_id2st.size(), "Input =", ids.size())); ClearCaches(); } else if (m_id2st.size() > ids.size() * 1.2) @@ -470,11 +928,11 @@ int HouseDetector::MergeStreets() { LOG(LDEBUG, ("MergeStreets() called", m_id2st.size())); -//#ifdef DEBUG -// KMLFileGuard file("dbg_merged_streets.kml"); -//#endif + //#ifdef DEBUG + // KMLFileGuard file("dbg_merged_streets.kml"); + //#endif - for (StreetMapT::iterator it = m_id2st.begin(); it != m_id2st.end(); ++it) + for (auto it = m_id2st.begin(); it != m_id2st.end(); ++it) { Street * st = it->second; @@ -488,15 +946,16 @@ int HouseDetector::MergeStreets() // Put longer streets first (for better house scoring). sort(m_streets.begin(), m_streets.end(), MergedStreet::GreaterLength()); -//#ifdef DEBUG -// char const * arrColor[] = { "FFFF0000", "FF00FFFF", "FFFFFF00", "FF0000FF", "FF00FF00", "FFFF00FF" }; + //#ifdef DEBUG + // char const * arrColor[] = { "FFFF0000", "FF00FFFF", "FFFFFF00", "FF0000FF", "FF00FF00", + // "FFFF00FF" }; -// // Write to kml from short to long to get the longest one at the top. -// for (int i = int(m_streets.size()) - 1; i >= 0; --i) -// { -// Streets2KML(file.GetStream(), m_streets[i], arrColor[i % ARRAY_SIZE(arrColor)]); -// } -//#endif + // // Write to kml from short to long to get the longest one at the top. + // for (int i = int(m_streets.size()) - 1; i >= 0; --i) + // { + // Streets2KML(file.GetStream(), m_streets[i], arrColor[i % ARRAY_SIZE(arrColor)]); + // } + //#endif LOG(LDEBUG, ("MergeStreets() result", m_streetNum)); return m_streetNum; @@ -545,8 +1004,10 @@ void MergedStreet::FinishReadingHouses() for (size_t i = 0; i < m_cont.size(); ++i) { if (i != 0) + { for (size_t j = 0; j < m_cont[i]->m_houses.size(); ++j) m_cont[i]->m_houses[j].m_streetDistance += length; + } length += m_cont[i]->m_length; m_cont[i]->m_housesRead = true; @@ -558,7 +1019,8 @@ void MergedStreet::FinishReadingHouses() HouseProjection const & p1 = Get(i); bool incI = true; - Index j = i; Inc(j); + Index j = i; + Inc(j); while (!IsEnd(j)) { HouseProjection const & p2 = Get(j); @@ -586,8 +1048,7 @@ void MergedStreet::FinishReadingHouses() HouseProjection const * MergedStreet::GetHousePivot(bool isOdd, bool & sign) const { - typedef my::limited_priority_queue< - HouseProjection const *, HouseProjection::LessDistance> QueueT; + typedef my::limited_priority_queue QueueT; QueueT q(HN_COUNT_FOR_ODD_TEST); // Get some most closest houses. @@ -596,7 +1057,7 @@ HouseProjection const * MergedStreet::GetHousePivot(bool isOdd, bool & sign) con // Calculate all probabilities. // even-left, odd-left, even-right, odd-right - double counter[4] = { 0, 0, 0, 0 }; + double counter[4] = {0, 0, 0, 0}; for (QueueT::const_iterator i = q.begin(); i != q.end(); ++i) { size_t ind = (*i)->m_house->GetIntNumber() % 2; @@ -625,18 +1086,19 @@ HouseProjection const * MergedStreet::GetHousePivot(bool isOdd, bool & sign) con return 0; } -template -void HouseDetector::ReadHouse(FeatureType const & f, Street * st, ProjectionCalcT & calc) +template +void HouseDetector::ReadHouse(FeatureType const & f, Street * st, ProjectionCalculator & calc) { string const houseNumber = f.GetHouseNumber(); /// @todo After new data generation we can skip IsHouseNumber check here. if (ftypes::IsBuildingChecker::Instance()(f) && feature::IsHouseNumber(houseNumber)) { - HouseMapT::iterator const it = m_id2house.find(f.GetID()); + auto const it = m_id2house.find(f.GetID()); bool const isNew = it == m_id2house.end(); - m2::PointD const pt = isNew ? f.GetLimitRect(FeatureType::BEST_GEOMETRY).Center() : it->second->GetPosition(); + m2::PointD const pt = + isNew ? f.GetLimitRect(FeatureType::BEST_GEOMETRY).Center() : it->second->GetPosition(); HouseProjection pr; if (calc.GetProjection(pt, pr) && pr.m_distMeters <= m_houseOffsetM) @@ -667,7 +1129,8 @@ void HouseDetector::ReadHouses(Street * st) if (st->m_housesRead) return; - //offsetMeters = max(HN_MIN_READ_OFFSET_M, min(GetApprLengthMeters(st->m_number) / 2, offsetMeters)); + // offsetMeters = max(HN_MIN_READ_OFFSET_M, min(GetApprLengthMeters(st->m_number) / 2, + // offsetMeters)); ProjectionOnStreetCalculator calc(st->m_points); m_loader.ForEachInRect(st->GetLimitRect(m_houseOffsetM), [this, &st, &calc](FeatureType & ft) { @@ -708,40 +1171,10 @@ void HouseDetector::ClearCaches() m_streets.clear(); } -namespace -{ - -class HasSecond -{ - set const & m_streets; -public: - HasSecond(set const & streets) : m_streets(streets) {} - template bool operator() (T const & t) const - { - return m_streets.count(t.second) > 0; - } -}; - -class HasStreet -{ - set const & m_streets; -public: - HasStreet(set const & streets) : m_streets(streets) {} - bool operator() (MergedStreet const & st) const - { - for (size_t i = 0; i < st.m_cont.size(); ++i) - if (m_streets.count(st.m_cont[i]) > 0) - return true; - return false; - } -}; - -} - void HouseDetector::ClearUnusedStreets(vector const & ids) { set streets; - for (StreetMapT::iterator it = m_id2st.begin(); it != m_id2st.end();) + for (auto it = m_id2st.begin(); it != m_id2st.end();) { if (!binary_search(ids.begin(), ids.end(), it->first)) { @@ -753,17 +1186,14 @@ void HouseDetector::ClearUnusedStreets(vector const & ids) } m_end2st.erase(remove_if(m_end2st.begin(), m_end2st.end(), HasSecond(streets)), m_end2st.end()); - m_streets.erase(remove_if(m_streets.begin(), m_streets.end(), HasStreet(streets)), m_streets.end()); + m_streets.erase(remove_if(m_streets.begin(), m_streets.end(), HasStreet(streets)), + m_streets.end()); for_each(streets.begin(), streets.end(), DeleteFunctor()); } -string DebugPrint(HouseProjection const & p) -{ - return p.m_house->GetNumber(); -} - -template void LogSequence(vector const & v) +template +void LogSequence(vector const & v) { #ifdef DEBUG for (size_t i = 0; i < v.size(); ++i) @@ -771,449 +1201,6 @@ template void LogSequence(vector const & v) #endif } -namespace -{ - -struct ScoredHouse -{ - House const * house; - double score; - ScoredHouse(House const * h, double s) : house(h), score(s) {} - ScoredHouse() : house(0), score(numeric_limits::max()) {} -}; - -class ResultAccumulator -{ - ParsedNumber m_number; - bool m_isOdd, m_sign, m_useOdd; - - ScoredHouse m_results[4]; - - bool IsBetter(int ind, double dist) const - { - return m_results[ind].house == 0 || m_results[ind].score > dist; - } - -public: - ResultAccumulator(string const & houseNumber) - : m_number(houseNumber) - { - } - - string const & GetFullNumber() const { return m_number.GetNumber(); } - bool UseOdd() const { return m_useOdd; } - - bool SetStreet(MergedStreet const & st) - { - Reset(); - - m_useOdd = true; - m_isOdd = m_number.IsOdd(); - return st.GetHousePivot(m_isOdd, m_sign) != 0; - } - - void SetSide(bool sign) - { - Reset(); - - m_useOdd = false; - m_sign = sign; - } - - void Reset() - { - for (size_t i = 0; i < ARRAY_SIZE(m_results); ++i) - m_results[i] = ScoredHouse(); - } - - void ResetNearby() { m_results[3] = ScoredHouse(); } - - bool IsOurSide(HouseProjection const & p) const - { - if (m_sign != p.m_projSign) - return false; - return (!m_useOdd || m_isOdd == p.IsOdd()); - } - - void ProcessCandidate(HouseProjection const & p) - { - if (IsOurSide(p)) - MatchCandidate(p, false); - } - - void MatchCandidate(HouseProjection const & p, bool checkNearby) - { - int ind = p.m_house->GetMatch(m_number); - if (ind == -1) - { - if (checkNearby && p.m_house->GetNearbyMatch(m_number)) - ind = 3; - else - return; - } - - if (IsBetter(ind, p.m_distMeters)) - m_results[ind] = ScoredHouse(p.m_house, p.m_distMeters); - } - - template - void FlushResults(TCont & cont) const - { - size_t const baseScore = 1 << (ARRAY_SIZE(m_results) - 2); - - for (size_t i = 0; i < ARRAY_SIZE(m_results) - 1; ++i) - if (m_results[i].house) - { - // Scores are: 4, 2, 1 according to the matching. - size_t const score = baseScore >> i; - - size_t j = 0; - for (; j < cont.size(); ++j) - if (cont[j].first == m_results[i].house) - { - cont[j].second += score; - break; - } - - if (j == cont.size()) - cont.push_back(make_pair(m_results[i].house, score)); - } - } - - House const * GetBestMatchHouse() const - { - return m_results[0].house; - } - - bool HasBestMatch() const { return (m_results[0].house != 0); } - House const * GetNearbyCandidate() const { return (HasBestMatch() ? 0 : m_results[3].house); } -}; -/* -void ProcessNearbyHouses(vector const & v, ResultAccumulator & acc) -{ - House const * p = acc.GetNearbyCandidate(); - if (p) - { - // Get additional search interval. - int const pivot = distance(v.begin(), find_if(v.begin(), v.end(), HouseProjection::EqualHouse(p))); - ASSERT(pivot >= 0 && pivot < v.size(), ()); - int const start = max(pivot - HN_NEARBY_INDEX_RANGE, 0); - int const end = min(pivot + HN_NEARBY_INDEX_RANGE + 1, int(v.size())); - - for (int i = start; i < end; ++i) - acc.MatchCandidate(*v[i], false); - } -} -*/ -//void GetClosestHouse(MergedStreet const & st, ResultAccumulator & acc) -//{ -// for (MergedStreet::Index i = st.Begin(); !st.IsEnd(i); st.Inc(i)) -// acc.ProcessCandidate(st.Get(i)); -//} - -// If it's odd or even part of the street pass 2, else pass 1. -void AddToQueue(int houseNumber, int step, queue & q) -{ - for (int i = 1; i <= 2; ++i) - { - int const delta = step * i; - q.push(houseNumber + delta); - if (houseNumber - delta > 0) - q.push(houseNumber - delta); - } -} - -struct HouseChain -{ - vector houses; - set chainHouses; - double score; - int minHouseNumber; - int maxHouseNumber; - - HouseChain() - { - minHouseNumber = -1; - maxHouseNumber = numeric_limits::max(); - } - - HouseChain(HouseProjection const * h) - { - minHouseNumber = maxHouseNumber = h->m_house->GetIntNumber(); - Add(h); - } - - void Add(HouseProjection const * h) - { - if (chainHouses.insert(h->m_house->GetNumber()).second) - { - int num = h->m_house->GetIntNumber(); - if (num < minHouseNumber) - minHouseNumber = num; - if (num > maxHouseNumber) - maxHouseNumber = num; - houses.push_back(h); - } - } - - bool Find(string const & str) - { - return (chainHouses.find(str) != chainHouses.end()); - } - - void CountScore() - { - sort(houses.begin(), houses.end(), HouseProjection::LessDistance()); - size_t const size = houses.size(); - score = 0; - size_t const scoreNumber = 3; - for (size_t i = 0; i < scoreNumber; ++i) - score += i < size ? houses[i]->m_distMeters : houses.back()->m_distMeters; - score /= scoreNumber; - } - - bool IsIntersecting(HouseChain const & chain) const - { - if (minHouseNumber >= chain.maxHouseNumber) - return false; - if (chain.minHouseNumber >= maxHouseNumber) - return false; - return true; - } - - bool operator<(HouseChain const & p) const - { - return score < p.score; - } -}; - -void GetBestHouseFromChains(vector & houseChains, ResultAccumulator & acc) -{ - for (size_t i = 0; i < houseChains.size(); ++i) - houseChains[i].CountScore(); - sort(houseChains.begin(), houseChains.end()); - - ParsedNumber number(acc.GetFullNumber()); - - for (size_t i = 0; i < houseChains.size(); ++i) - { - if (i == 0 || !houseChains[0].IsIntersecting(houseChains[i])) - { - for (size_t j = 0; j < houseChains[i].houses.size(); ++j) - { - if (houseChains[i].houses[j]->m_house->GetMatch(number) != -1) - { - for (size_t k = 0; k < houseChains[i].houses.size(); ++k) - acc.MatchCandidate(*houseChains[i].houses[k], true); - break; - } - } - } - } -} - - -struct Competitiors -{ - uint32_t m_candidateIndex; - uint32_t m_chainIndex; - double m_score; - Competitiors(uint32_t candidateIndex, uint32_t chainIndex, double score) - : m_candidateIndex(candidateIndex), m_chainIndex(chainIndex), m_score(score) - {} - bool operator<(Competitiors const & c) const - { - return m_score < c.m_score; - } -}; - - -void ProccessHouses(vector const & st, ResultAccumulator & acc) -{ - vector houseChains; - size_t const count = st.size(); - size_t numberOfStreetHouses = count; - vector used(count, false); - string const & houseNumber = acc.GetFullNumber(); - int const step = acc.UseOdd() ? 2 : 1; - - for (size_t i = 0; i < count; ++i) - { - HouseProjection const * hp = st[i]; - if (st[i]->m_house->GetNumber() == houseNumber) - { - houseChains.push_back(HouseChain(hp)); - used[i] = true; - --numberOfStreetHouses; - } - } - if (houseChains.empty()) - return; - - queue houseNumbersToCheck; - AddToQueue(houseChains[0].houses[0]->m_house->GetIntNumber(), step, houseNumbersToCheck); - while (numberOfStreetHouses > 0) - { - if (!houseNumbersToCheck.empty()) - { - int candidateHouseNumber = houseNumbersToCheck.front(); - houseNumbersToCheck.pop(); - vector candidates; - ASSERT_LESS(used.size(), numeric_limits::max(), ()); - uint32_t const count = static_cast(used.size()); - for (uint32_t i = 0; i < count; ++i) - if (!used[i] && st[i]->m_house->GetIntNumber() == candidateHouseNumber) - candidates.push_back(i); - - bool shouldAddHouseToQueue = false; - vector comp; - - for (size_t i = 0; i < candidates.size(); ++i) - { - string num = st[candidates[i]]->m_house->GetNumber(); - ASSERT_LESS(houseChains.size(), numeric_limits::max(), ()); - for (size_t j = 0; j < houseChains.size(); ++j) - { - if (!houseChains[j].Find(num)) - { - double dist = numeric_limits::max(); - for (size_t k = 0; k < houseChains[j].houses.size(); ++k) - { - if (abs(houseChains[j].houses[k]->m_house->GetIntNumber() - st[candidates[i]]->m_house->GetIntNumber()) <= HN_NEARBY_DISTANCE) - dist = min(dist, GetDistanceMeters(houseChains[j].houses[k]->m_house->GetPosition(), st[candidates[i]]->m_house->GetPosition())); - } - if (dist < HN_MAX_CONNECTION_DIST_M) - comp.push_back(Competitiors(candidates[i], static_cast(j), dist)); - } - } - } - sort(comp.begin(), comp.end()); - - for (size_t i = 0; i < comp.size(); ++i) - { - if (!used[comp[i].m_candidateIndex]) - { - string num = st[comp[i].m_candidateIndex]->m_house->GetNumber(); - if (!houseChains[comp[i].m_chainIndex].Find(num)) - { - used[comp[i].m_candidateIndex] = true; - houseChains[comp[i].m_chainIndex].Add(st[comp[i].m_candidateIndex]); - --numberOfStreetHouses; - shouldAddHouseToQueue = true; - } - } - } - if (shouldAddHouseToQueue) - AddToQueue(candidateHouseNumber, step, houseNumbersToCheck); - } - else - { - for (size_t i = 0; i < used.size(); ++i) - { - if (!used[i]) - { - houseChains.push_back(HouseChain(st[i])); - --numberOfStreetHouses; - used[i] = true; - AddToQueue(st[i]->m_house->GetIntNumber(), step, houseNumbersToCheck); - break; - } - } - } - } - GetBestHouseFromChains(houseChains, acc); -} - -void GetBestHouseWithNumber(MergedStreet const & st, double offsetMeters, ResultAccumulator & acc) -{ - vector v; - for (MergedStreet::Index i = st.Begin(); !st.IsEnd(i); st.Inc(i)) - { - HouseProjection const & p = st.Get(i); - if (p.m_distMeters <= offsetMeters && acc.IsOurSide(p)) - v.push_back(&p); - } - - ProccessHouses(v, acc); -} - -struct CompareHouseNumber -{ - inline bool Less(HouseProjection const * h1, HouseProjection const * h2) const - { - return (h1->m_house->GetIntNumber() <= h2->m_house->GetIntNumber()); - } - inline bool Greater(HouseProjection const * h1, HouseProjection const * h2) const - { - return (h1->m_house->GetIntNumber() >= h2->m_house->GetIntNumber()); - } -}; -/* -void LongestSubsequence(vector const & v, - vector & res) -{ - LongestSubsequence(v, back_inserter(res), CompareHouseNumber()); -} -*/ -//void GetLSHouse(MergedStreet const & st, double offsetMeters, ResultAccumulator & acc) -//{ -// acc.ResetNearby(); - -// vector v; -// for (MergedStreet::Index i = st.Begin(); !st.IsEnd(i); st.Inc(i)) -// { -// search::HouseProjection const & p = st.Get(i); -// if (p.m_distance <= offsetMeters && acc.IsOurSide(p)) -// v.push_back(&p); -// } - -// vector res; -// LongestSubsequence(v, res); - -// //LOG(LDEBUG, ("=== Offset", offsetMeters, "===")); -// //LogSequence(res); - -// for (size_t i = 0; i < res.size(); ++i) -// acc.MatchCandidate(*(res[i]), true); - -// ProcessNearbyHouses(v, acc); -//} - -struct GreaterSecond -{ - template - bool operator() (T const & t1, T const & t2) const { return t1.second > t2.second; } -}; - -void ProduceVoting(vector const & acc, - vector & res, - MergedStreet const & st) -{ - buffer_vector, 4> voting; - - // Calculate score for every house. - for (size_t i = 0; i < acc.size(); ++i) - acc[i].FlushResults(voting); - - if (voting.empty()) - return; - - // Sort according to the score (bigger is better). - sort(voting.begin(), voting.end(), GreaterSecond()); - - // Emit results with equal best score. - size_t const score = voting[0].second; - for (size_t i = 0; i < voting.size(); ++i) - { - if (score == voting[i].second) - res.push_back(HouseResult(voting[i].first, &st)); - else - break; - } -} - -} - void HouseDetector::GetHouseForName(string const & houseNumber, vector & res) { size_t const count = m_streets.size(); @@ -1235,12 +1222,14 @@ void HouseDetector::GetHouseForName(string const & houseNumber, vectorGetNumber(); } + +string DebugPrint(HouseResult const & r) +{ + return r.m_house->GetNumber() + ", " + r.m_street->GetName(); } +} // namespace search diff --git a/search/house_detector.hpp b/search/house_detector.hpp index 8e72cdb43b..c1d71a270c 100644 --- a/search/house_detector.hpp +++ b/search/house_detector.hpp @@ -10,8 +10,11 @@ #include "base/macros.hpp" -#include "std/string.hpp" -#include "std/queue.hpp" +#include +#include +#include +#include +#include class DataSourceBase; @@ -19,34 +22,30 @@ namespace search { struct ParsedNumber { - string m_fullN; - int m_startN, m_endN; - public: /// @todo Pass correct "American" notation flag. - ParsedNumber(string const & number, bool american = false); + ParsedNumber(std::string const & number, bool american = false); - inline string const & GetNumber() const { return m_fullN; } - inline bool IsOdd() const { return (m_startN % 2 == 1); } - inline int GetIntNumber() const { return m_startN; } + std::string const & GetNumber() const { return m_fullN; } + bool IsOdd() const { return (m_startN % 2 == 1); } + int GetIntNumber() const { return m_startN; } bool IsIntersect(ParsedNumber const & number, int offset = 0) const; + +private: + std::string m_fullN; + int m_startN; + int m_endN; }; class House { - ParsedNumber m_number; - m2::PointD m_point; - public: - House(string const & number, m2::PointD const & point) - : m_number(number), m_point(point) - { - } + House(std::string const & number, m2::PointD const & point) : m_number(number), m_point(point) {} - inline string const & GetNumber() const { return m_number.GetNumber(); } - inline int GetIntNumber() const { return m_number.GetIntNumber(); } - inline m2::PointD const & GetPosition() const { return m_point; } + std::string const & GetNumber() const { return m_number.GetNumber(); } + int GetIntNumber() const { return m_number.GetIntNumber(); } + m2::PointD const & GetPosition() const { return m_point; } /// @return \n /// -1 - no match; @@ -55,22 +54,19 @@ public: /// 2 - integer number match. int GetMatch(ParsedNumber const & number) const; bool GetNearbyMatch(ParsedNumber const & number) const; + +private: + ParsedNumber m_number; + m2::PointD m_point; }; // NOTE: DO NOT DELETE instances of this class by a pointer/reference // to ProjectionOnStreet, because both classes have non-virtual destructors. struct HouseProjection : public ProjectionOnStreet { - House const * m_house; - - /// Distance in mercator, from street beginning to projection on street - double m_streetDistance; - - inline bool IsOdd() const { return (m_house->GetIntNumber() % 2 == 1); } - struct LessDistance { - bool operator() (HouseProjection const * p1, HouseProjection const * p2) const + bool operator()(HouseProjection const * p1, HouseProjection const * p2) const { return p1->m_distMeters < p2->m_distMeters; } @@ -78,28 +74,26 @@ struct HouseProjection : public ProjectionOnStreet class EqualHouse { - House const * m_house; public: EqualHouse(House const * h) : m_house(h) {} bool operator() (HouseProjection const * p) const { return m_house == p->m_house; } + + private: + House const * m_house; }; + + bool IsOdd() const { return (m_house->GetIntNumber() % 2 == 1); } + + House const * m_house; + + /// Distance in mercator, from street beginning to projection on street + double m_streetDistance; }; // many features combines to street class Street { - string m_name; - string m_processedName; - public: - void SetName(string const & name); - - vector m_points; - vector m_houses; - double m_length; /// Length in mercator - int m_number; /// Some ordered number after merging - bool m_housesRead; - Street() : m_length(0.0), m_number(-1), m_housesRead(false) {} void Reverse(); @@ -112,29 +106,34 @@ public: double GetPrefixLength(size_t numSegs) const; - inline static bool IsSameStreets(Street const * s1, Street const * s2) + static bool IsSameStreets(Street const * s1, Street const * s2) { return s1->m_processedName == s2->m_processedName; } - inline string const & GetDbgName() const { return m_processedName; } - inline string const & GetName() const { return m_name; } + void SetName(std::string const & name); + std::string const & GetDbgName() const { return m_processedName; } + std::string const & GetName() const { return m_name; } + + std::vector m_points; + std::vector m_houses; + double m_length; /// Length in mercator + int m_number; /// Some ordered number after merging + bool m_housesRead; + +private: + std::string m_name; + std::string m_processedName; }; class MergedStreet { - double m_length; public: - deque m_cont; - - MergedStreet() : m_length(0.0) {} - - string const & GetDbgName() const; - string const & GetName() const; - bool IsHousesRead() const; - void FinishReadingHouses(); - - HouseProjection const * GetHousePivot(bool isOdd, bool & sign) const; + struct Index + { + size_t s, h; + Index() : s(0), h(0) {} + }; struct GreaterLength { @@ -144,113 +143,116 @@ public: } }; - inline void Swap(MergedStreet & s) + MergedStreet() : m_length(0.0) {} + + std::string const & GetDbgName() const; + std::string const & GetName() const; + bool IsHousesRead() const; + void FinishReadingHouses(); + + HouseProjection const * GetHousePivot(bool isOdd, bool & sign) const; + + void Swap(MergedStreet & s) { m_cont.swap(s.m_cont); std::swap(m_length, s.m_length); } -public: - struct Index - { - size_t s, h; - Index() : s(0), h(0) {} - }; - - inline Index Begin() const + Index Begin() const { Index i; Next(i); return i; } - inline void Inc(Index & i) const + + void Inc(Index & i) const { ++i.h; Next(i); } - inline bool IsEnd(Index const & i) const - { - return i.s == m_cont.size(); - } - inline HouseProjection const & Get(Index const & i) const + + bool IsEnd(Index const & i) const { return i.s == m_cont.size(); } + + HouseProjection const & Get(Index const & i) const { ASSERT(!IsEnd(i), ()); return m_cont[i.s]->m_houses[i.h]; } + std::deque m_cont; + private: void Erase(Index & i); void Next(Index & i) const; -}; -inline void swap(MergedStreet & s1, MergedStreet & s2) -{ - s1.Swap(s2); -} + double m_length; +}; struct HouseResult { - House const * m_house; - MergedStreet const * m_street; - HouseResult(House const * house, MergedStreet const * street) : m_house(house), m_street(street) { } - inline bool operator<(HouseResult const & a) const { return m_house < a.m_house; } - inline bool operator==(HouseResult const & a) const { return m_house == a.m_house; } + bool operator<(HouseResult const & a) const { return m_house < a.m_house; } + bool operator==(HouseResult const & a) const { return m_house == a.m_house; } m2::PointD const & GetOrg() const { return m_house->GetPosition(); } -}; -inline string DebugPrint(HouseResult const & r) -{ - return r.m_house->GetNumber() + ", " + r.m_street->GetName(); -} + House const * m_house; + MergedStreet const * m_street; +}; class HouseDetector { - FeatureLoader m_loader; +public: + using StreetMap = std::map; + using HouseMap = std::map; + using StreetPtr = std::pair; - typedef map StreetMapT; - StreetMapT m_id2st; - typedef map HouseMapT; - HouseMapT m_id2house; + static int const DEFAULT_OFFSET_M = 200; - vector > m_end2st; - vector m_streets; + HouseDetector(DataSourceBase const & dataSource); + ~HouseDetector(); - double m_metres2Mercator; - int m_streetNum; - double m_houseOffsetM; + int LoadStreets(std::vector const & ids); + /// @return number of different joined streets. + int MergeStreets(); - typedef pair StreetPtr; + void ReadAllHouses(double offsetMeters = DEFAULT_OFFSET_M); + + void GetHouseForName(std::string const & houseNumber, std::vector & res); + + void ClearCaches(); + void ClearUnusedStreets(std::vector const & ids); + +private: StreetPtr FindConnection(Street const * st, bool beg) const; + void MergeStreets(Street * st); - template - void ReadHouse(FeatureType const & f, Street * st, ProjectionCalcT & calc); + template + void ReadHouse(FeatureType const & f, Street * st, ProjectionCalculator & calc); + void ReadHouses(Street * st); void SetMetres2Mercator(double factor); double GetApprLengthMeters(int index) const; -public: - HouseDetector(DataSourceBase const & dataSource); - ~HouseDetector(); + FeatureLoader m_loader; - int LoadStreets(vector const & ids); - /// @return number of different joined streets. - int MergeStreets(); + StreetMap m_id2st; + HouseMap m_id2house; - static int const DEFAULT_OFFSET_M = 200; - void ReadAllHouses(double offsetMeters = DEFAULT_OFFSET_M); + std::vector> m_end2st; + std::vector m_streets; - void GetHouseForName(string const & houseNumber, vector & res); - - void ClearCaches(); - void ClearUnusedStreets(vector const & ids); + double m_metres2Mercator; + int m_streetNum; + double m_houseOffsetM; }; -} +std::string DebugPrint(HouseProjection const & p); +std::string DebugPrint(HouseResult const & r); +} // namespace search