[search] Built-in house search in main engine (initial commit).

This commit is contained in:
vng 2014-01-08 19:15:08 +03:00 committed by Alex Zolotarev
parent b00f54d0cc
commit 097e071f80
7 changed files with 137 additions and 17 deletions

View file

@ -183,10 +183,11 @@ void HouseDetector::Bfs(Street * st)
}
}
void HouseDetector::LoadStreets(vector<FeatureID> & ids)
int HouseDetector::LoadStreets(vector<FeatureID> & ids)
{
sort(ids.begin(), ids.end());
int count = 0;
for (size_t i = 0; i < ids.size(); ++i)
{
if (m_id2st.find(ids[i]) != m_id2st.end())
@ -201,6 +202,8 @@ void HouseDetector::LoadStreets(vector<FeatureID> & ids)
if (!f.GetName(0, name) || name.empty())
continue;
++count;
Street * st = new Street();
st->SetName(name);
f.ForEachPoint(StreetCreator(st), FeatureType::BEST_GEOMETRY);
@ -214,6 +217,7 @@ void HouseDetector::LoadStreets(vector<FeatureID> & ids)
}
m_loader.Free();
return count;
}
int HouseDetector::MergeStreets()
@ -407,6 +411,10 @@ void HouseDetector::MatchAllHouses(string const & houseNumber, vector<HouseProje
}
}
void HouseDetector::GetHouseForName(string const & houseNumber, vector<House> & res)
{
}
namespace
@ -558,7 +566,8 @@ void ProccessHouses(vector <search::HouseProjection> & houses, vector <search::H
}
//valid only if only one street in class
void GetAllHousesForStreet(pair <search::HouseDetector::IterM, search::HouseDetector::IterM> range, map<search::House, double> & m)
void GetAllHousesForStreet(pair<search::HouseDetector::IterM, search::HouseDetector::IterM> range,
map<search::House, double> & m)
{
for (search::HouseDetector::IterM it = range.first; it != range.second; ++it)
{

View file

@ -150,7 +150,7 @@ public:
HouseDetector(Index const * pIndex);
void LoadStreets(vector<FeatureID> & ids);
int LoadStreets(vector<FeatureID> & ids);
/// @return number of different joined streets.
int MergeStreets();
@ -158,7 +158,7 @@ public:
void ReadAllHouses(double offsetMeters);
void MatchAllHouses(string const & houseNumber, vector<HouseProjection> & res);
vector<House> GetHouseForName(string const & houseName);
void GetHouseForName(string const & houseNumber, vector<House> & res);
pair<IterM, IterM> GetAllStreets();
void SimpleFilter(string const & houseNumber, vector<House> & res);
@ -166,6 +166,7 @@ public:
void LongestSubsequence(vector<HouseProjection> const & houses,
vector<HouseProjection> & result);
void GetAllHousesForStreet(pair <search::HouseDetector::IterM, search::HouseDetector::IterM> range, map<House, double> & m);
void GetAllHousesForStreet(pair<search::HouseDetector::IterM, search::HouseDetector::IterM> range,
map<House, double> & m);
}

View file

@ -336,6 +336,44 @@ namespace
return (find(m_index, e, FirstLevelIndex(type)) != e);
}
};
class IsStreetChecker
{
vector<uint32_t> m_types;
public:
IsStreetChecker()
{
Classificator const & c = classif();
char const * arr[][2] = {
{ "highway", "primary" },
{ "highway", "secondary" },
{ "highway", "residential" },
{ "highway", "pedestrian" },
{ "highway", "residential" },
{ "highway", "tertiary" },
{ "highway", "construction" },
{ "highway", "living_street" },
{ "highway", "trunk" }
};
for (size_t i = 0; i < ARRAY_SIZE(arr); ++i)
m_types.push_back(c.GetTypeByPath(vector<string>(arr[i], arr[i] + 2)));
}
bool IsMy(feature::TypesHolder const & types) const
{
for (size_t i = 0; i < types.Size(); ++i)
{
uint32_t t = types[i];
ftype::TruncValue(t, 2);
if (find(m_types.begin(), m_types.end(), t) != m_types.end())
return true;
}
return false;
}
};
}
bool PreResult2::LessLinearTypesF::operator() (PreResult2 const & r1, PreResult2 const & r2) const
@ -375,6 +413,12 @@ bool PreResult2::EqualLinearTypesF::operator() (PreResult2 const & r1, PreResult
return false;
}
bool PreResult2::IsStreet() const
{
static IsStreetChecker checker;
return checker.IsMy(m_types);
}
string PreResult2::DebugPrint() const
{
string res("IntermediateResult: ");

View file

@ -119,6 +119,9 @@ public:
string DebugPrint() const;
bool IsStreet() const;
inline FeatureID const & GetID() const { return m_id; }
private:
template <class T> friend bool LessRankT(T const & r1, T const & r2);
template <class T> friend bool LessViewportDistanceT(T const & r1, T const & r2);

View file

@ -85,7 +85,7 @@ Engine::Engine(IndexType const * pIndex, Reader * pCategoriesR,
&m_pData->m_categories,
&m_pData->m_stringsToSuggest,
&m_pData->m_infoGetter,
RESULTS_COUNT));
100)); // Temporary for houses test
m_pQuery->SetPreferredLanguage(lang);
}

View file

@ -75,6 +75,7 @@ Query::Query(Index const * pIndex,
m_pCategories(pCategories),
m_pStringsToSuggest(pStringsToSuggest),
m_pInfoGetter(pInfoGetter),
m_houseDetector(pIndex),
m_worldSearch(true),
m_position(empty_pos_value, empty_pos_value)
{
@ -266,6 +267,11 @@ void Query::ClearQueues()
void Query::SetQuery(string const & query)
{
#ifdef HOUSE_SEARCH_TEST
m_house.clear();
m_streetID.clear();
#endif
// split input query by tokens and prefix
search::Delimiters delims;
SplitUniString(NormalizeAndSimplifyString(query), MakeBackInsertFunctor(m_tokens), delims);
@ -295,6 +301,31 @@ void Query::SearchCoordinates(string const & query, Results & res) const
}
}
namespace
{
bool IsNumber(strings::UniString const & s)
{
for (size_t i = 0; i < s.size(); ++i)
{
// android production ndk-r8d has bug. "еда" detected as a number.
if (s[i] > 127 || !isdigit(s[i]))
return false;
}
return true;
}
bool isHouseNumber(strings::UniString const & s)
{
if (s.size() > 6)
return false;
if (s[0] > 127 || !isdigit(s[0]))
return false;
cout << strings::ToUtf8(s) << " is house number" << endl;
return true;
}
}
void Query::Search(Results & res, bool searchAddress)
{
ClearQueues();
@ -302,6 +333,14 @@ void Query::Search(Results & res, bool searchAddress)
if (m_cancel) return;
SuggestStrings(res);
#ifdef HOUSE_SEARCH_TEST
if (m_tokens.size() > 1 && isHouseNumber(m_tokens.back()))
{
m_house.swap(m_tokens.back());
m_tokens.pop_back();
}
#endif
if (m_cancel) return;
if (searchAddress)
SearchAddress();
@ -509,6 +548,28 @@ void Query::FlushResults(Results & res, void (Results::*pAddFn)(Result const &))
if (indV.empty())
return;
#ifdef HOUSE_SEARCH_TEST
if (!m_house.empty())
{
vector<FeatureID> streets;
for (size_t i = 0; i < indV.size(); ++i)
{
impl::PreResult2 const & r = *(indV[i]);
if (r.IsStreet())
streets.push_back(r.GetID());
}
if (m_houseDetector.LoadStreets(streets) > 0)
m_houseDetector.MergeStreets();
vector<search::House> houses;
m_houseDetector.GetHouseForName(strings::ToUtf8(m_house), houses);
for (size_t i = 0; i < houses.size(); ++i)
(res.*pAddFn)(Result(houses[i].GetPosition(), houses[i].GetNumber(),
string(), string(), houses[i].GetPosition().Length(m_position)));
}
#endif
RemoveDuplicatingLinear(indV);
for (size_t i = 0; i < m_qCount; ++i)
@ -920,17 +981,6 @@ string DebugPrint(Query::Params const & p)
namespace
{
bool IsNumber(strings::UniString const & s)
{
for (size_t i = 0; i < s.size(); ++i)
{
// android production ndk-r8d has bug. "еда" detected as a number.
if (s[i] > 127 || !isdigit(s[i]))
return false;
}
return true;
}
class DoStoreNumbers
{
vector<size_t> & m_vec;

View file

@ -18,6 +18,13 @@
#include "../std/vector.hpp"
#define HOUSE_SEARCH_TEST
#ifdef HOUSE_SEARCH_TEST
#include "house_detector.hpp"
#endif
class FeatureType;
class CategoriesHolder;
@ -190,6 +197,12 @@ private:
buffer_vector<strings::UniString, 32> m_tokens;
strings::UniString m_prefix;
#ifdef HOUSE_SEARCH_TEST
strings::UniString m_house;
vector<FeatureID> m_streetID;
search::HouseDetector m_houseDetector;
#endif
/// 0 - current viewport rect
/// 1 - near me rect
/// 2 - around city rect