forked from organicmaps/organicmaps
Knee-search v0.2, limited to viewport only
This commit is contained in:
parent
1d70a6e803
commit
3b06c903f7
8 changed files with 190 additions and 44 deletions
|
@ -202,6 +202,11 @@ public:
|
|||
return m_Params.layer;
|
||||
}
|
||||
|
||||
inline bool HasName() const
|
||||
{
|
||||
return Header() & feature::HEADER_HAS_NAME;
|
||||
}
|
||||
|
||||
template <class T>
|
||||
inline bool ForEachNameRef(T & functor) const
|
||||
{
|
||||
|
|
|
@ -6,6 +6,7 @@
|
|||
#include "feature_vec_model.hpp"
|
||||
#include "benchmark_provider.hpp"
|
||||
#include "languages.hpp"
|
||||
#include "search_processor.hpp"
|
||||
|
||||
#include "../indexer/feature_visibility.hpp"
|
||||
#include "../indexer/feature.hpp"
|
||||
|
@ -257,7 +258,6 @@ namespace fwork
|
|||
|
||||
return true;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
template <typename TModel>
|
||||
|
@ -387,7 +387,7 @@ void FrameWork<TModel>::AddRedrawCommandSure()
|
|||
// set language priorities
|
||||
languages::CodesT langCodes;
|
||||
languages::GetCurrentSettings(langCodes);
|
||||
StringUtf8Multilang::SetPreferableLanguages(langCodes);
|
||||
languages::SaveSettings(langCodes);
|
||||
}
|
||||
|
||||
template <typename TModel>
|
||||
|
@ -1058,37 +1058,21 @@ void FrameWork<TModel>::AddRedrawCommandSure()
|
|||
UpdateNow();
|
||||
}
|
||||
|
||||
class SearchProcessor
|
||||
{
|
||||
string const & m_text;
|
||||
SearchCallbackT & m_callback;
|
||||
|
||||
public:
|
||||
SearchProcessor(string const & textToSearch, SearchCallbackT & callback)
|
||||
: m_text(textToSearch), m_callback(callback) {}
|
||||
bool operator() (FeatureType const & f) const
|
||||
{
|
||||
// @TODO search for all languages
|
||||
string name;
|
||||
f.GetName(name);
|
||||
if (!name.empty() && name.find(m_text) != string::npos)
|
||||
{
|
||||
m_callback(name, f.GetLimitRect(-1));
|
||||
}
|
||||
return true;
|
||||
}
|
||||
};
|
||||
|
||||
template<typename TModel>
|
||||
void FrameWork<TModel>::Search(string const & text, SearchCallbackT callback) const
|
||||
{
|
||||
threads::MutexGuard lock(m_modelSyn);
|
||||
|
||||
SearchProcessor doClass(text, callback);
|
||||
m_model.ForEachFeatureWithScale(m2::RectD(MercatorBounds::minX,
|
||||
search::Query query(text);
|
||||
search::Processor doClass(query);
|
||||
m_model.ForEachFeature(m_navigator.Screen().GlobalRect()
|
||||
/*m2::RectD(MercatorBounds::minX,
|
||||
MercatorBounds::minY,
|
||||
MercatorBounds::maxX,
|
||||
MercatorBounds::maxY), doClass, 9);
|
||||
MercatorBounds::maxY)*/, doClass);
|
||||
query.ForEachResultRef(callback);
|
||||
// empty name indicates last element
|
||||
callback(search::Result(string(), m2::RectD()));
|
||||
}
|
||||
|
||||
template class FrameWork<model::FeaturesFetcher>;
|
||||
|
|
|
@ -7,6 +7,7 @@
|
|||
#include "window_handle.hpp"
|
||||
#include "location_state.hpp"
|
||||
#include "navigator.hpp"
|
||||
#include "search_processor.hpp"
|
||||
|
||||
#include "../defines.hpp"
|
||||
|
||||
|
@ -51,7 +52,7 @@ class redraw_operation_cancelled {};
|
|||
|
||||
struct BenchmarkRectProvider;
|
||||
|
||||
typedef boost::function<void (string const &, m2::RectD const &)> SearchCallbackT;
|
||||
typedef boost::function<void (search::Result const &)> SearchCallbackT;
|
||||
|
||||
namespace fwork
|
||||
{
|
||||
|
|
|
@ -25,6 +25,7 @@ HEADERS += \
|
|||
location_state.hpp \
|
||||
benchmark_provider.hpp \
|
||||
languages.hpp \
|
||||
search_processor.hpp \
|
||||
|
||||
SOURCES += \
|
||||
feature_vec_model.cpp \
|
||||
|
@ -39,6 +40,7 @@ SOURCES += \
|
|||
location_state.cpp \
|
||||
benchmark_provider.cpp \
|
||||
languages.cpp \
|
||||
search_processor.cpp \
|
||||
|
||||
!iphone*:!bada* {
|
||||
HEADERS += qgl_render_context.hpp
|
||||
|
|
79
map/search_processor.cpp
Normal file
79
map/search_processor.cpp
Normal file
|
@ -0,0 +1,79 @@
|
|||
#include "search_processor.hpp"
|
||||
|
||||
#include "../indexer/feature.hpp"
|
||||
#include "../indexer/classificator.hpp"
|
||||
|
||||
#include "../base/utf8_string.hpp"
|
||||
#include "../base/logging.hpp"
|
||||
|
||||
#include "../std/bind.hpp"
|
||||
|
||||
namespace search
|
||||
{
|
||||
int Score(string const & key, string const & word)
|
||||
{
|
||||
size_t const offset = word.find(key);
|
||||
switch (offset)
|
||||
{
|
||||
case 0: // best match - from the beginning
|
||||
if (word.size() == key.size())
|
||||
return 1000; // full match
|
||||
else
|
||||
return 100; // partial match
|
||||
break;
|
||||
case string::npos: // no match
|
||||
return -1;
|
||||
break;
|
||||
default: // match in the middle of the string
|
||||
return key.size() * 2; // how many symbols matched
|
||||
}
|
||||
}
|
||||
|
||||
Query::Query(string const & line)
|
||||
{
|
||||
utf8_string::Split(line, m_tokens);
|
||||
}
|
||||
|
||||
bool Query::operator()(char lang, string const & utf8s)
|
||||
{
|
||||
vector<string> words;
|
||||
utf8_string::Split(utf8s, words);
|
||||
int score = -1;
|
||||
for (size_t i = 0; i < m_tokens.size(); ++i)
|
||||
{
|
||||
for (size_t j = 0; j < words.size(); ++j)
|
||||
{
|
||||
score += Score(m_tokens[i], words[j]);
|
||||
}
|
||||
}
|
||||
if (score > 0)
|
||||
{
|
||||
m_queue.push(make_pair(score, Result(utf8s, m_currFeature->GetLimitRect(-1))));
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
void Query::Match(FeatureType const & f)
|
||||
{
|
||||
m_currFeature = &f;
|
||||
f.ForEachNameRef(*this);
|
||||
}
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
Processor::Processor(Query & query)
|
||||
: m_query(query)
|
||||
{
|
||||
}
|
||||
|
||||
bool Processor::operator() (FeatureType const & f) const
|
||||
{
|
||||
// filter out features without any name
|
||||
if (f.HasName())
|
||||
m_query.Match(f);
|
||||
return true;
|
||||
}
|
||||
|
||||
}
|
65
map/search_processor.hpp
Normal file
65
map/search_processor.hpp
Normal file
|
@ -0,0 +1,65 @@
|
|||
#pragma once
|
||||
|
||||
#include "../geometry/rect2d.hpp"
|
||||
|
||||
#include "../std/string.hpp"
|
||||
#include "../std/vector.hpp"
|
||||
#include "../std/queue.hpp"
|
||||
|
||||
class FeatureType;
|
||||
|
||||
namespace search
|
||||
{
|
||||
struct Result
|
||||
{
|
||||
Result(string const & name, m2::RectD const & rect)
|
||||
: m_name(name), m_rect(rect) {}
|
||||
string m_name;
|
||||
m2::RectD m_rect;
|
||||
};
|
||||
|
||||
typedef pair<int, search::Result> elem_type;
|
||||
struct QueueComparer
|
||||
{
|
||||
// custom comparison for priority_queue
|
||||
bool operator() (elem_type const & e1, elem_type const & e2) const
|
||||
{
|
||||
return e1.first < e2.first;
|
||||
}
|
||||
};
|
||||
|
||||
class Query
|
||||
{
|
||||
vector<string> m_tokens;
|
||||
/// custom comparison, holds found features
|
||||
priority_queue<elem_type, vector<elem_type>, QueueComparer> m_queue;
|
||||
|
||||
// @TODO refactor and remove
|
||||
FeatureType const * m_currFeature;
|
||||
|
||||
public:
|
||||
Query(string const & line);
|
||||
|
||||
bool operator()(char lang, string const & utf8s);
|
||||
|
||||
void Match(FeatureType const & f);
|
||||
template <class T> void ForEachResultRef(T & f)
|
||||
{
|
||||
while (!m_queue.empty())
|
||||
{
|
||||
f(m_queue.top().second);
|
||||
m_queue.pop();
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
class Processor
|
||||
{
|
||||
/// mutable here because indexer stores const ref to functor
|
||||
mutable Query & m_query;
|
||||
|
||||
public:
|
||||
Processor(Query & query);
|
||||
bool operator() (FeatureType const & f) const;
|
||||
};
|
||||
}
|
|
@ -6,6 +6,7 @@
|
|||
|
||||
#include "../defines.hpp"
|
||||
|
||||
#include "../map/search_processor.hpp"
|
||||
#include "../map/settings.hpp"
|
||||
|
||||
#include <QtGui/QDockWidget>
|
||||
|
@ -349,27 +350,31 @@ void MainWindow::OnSearchTextChanged(QString const & str)
|
|||
QTableWidget * table = static_cast<QTableWidget *>(m_Docks[3]->widget());
|
||||
table->clear();
|
||||
table->setRowCount(0);
|
||||
m_pDrawWidget->Search(str.toUtf8().constData(),
|
||||
boost::bind(&MainWindow::OnSearchResult, this, _1, _2));
|
||||
if (!str.isEmpty())
|
||||
m_pDrawWidget->Search(str.toUtf8().constData(),
|
||||
boost::bind(&MainWindow::OnSearchResult, this, _1));
|
||||
}
|
||||
|
||||
void MainWindow::OnSearchResult(string const & name, m2::RectD const & rect)
|
||||
void MainWindow::OnSearchResult(search::Result const & result)
|
||||
{
|
||||
QTableWidget * table = static_cast<QTableWidget *>(m_Docks[3]->widget());
|
||||
if (result.m_name.empty()) // last element
|
||||
{
|
||||
if (!m_Docks[3]->isVisible())
|
||||
m_Docks[3]->show();
|
||||
}
|
||||
else
|
||||
{
|
||||
QTableWidget * table = static_cast<QTableWidget *>(m_Docks[3]->widget());
|
||||
|
||||
int const rowCount = table->rowCount();
|
||||
if (rowCount > 100)
|
||||
return;
|
||||
int const rowCount = table->rowCount();
|
||||
|
||||
table->setRowCount(rowCount + 1);
|
||||
QTableWidgetItem * item = new QTableWidgetItem(QString::fromUtf8(name.c_str()));
|
||||
item->setData(Qt::UserRole, QRectF(QPointF(rect.minX(), rect.maxY()),
|
||||
QPointF(rect.maxX(), rect.minY())));
|
||||
item->setFlags(Qt::ItemIsSelectable | Qt::ItemIsEnabled);
|
||||
table->setItem(rowCount, 0, item);
|
||||
|
||||
if (!m_Docks[3]->isVisible())
|
||||
m_Docks[3]->show();
|
||||
table->setRowCount(rowCount + 1);
|
||||
QTableWidgetItem * item = new QTableWidgetItem(QString::fromUtf8(result.m_name.c_str()));
|
||||
item->setData(Qt::UserRole, QRectF(QPointF(result.m_rect.minX(), result.m_rect.maxY()),
|
||||
QPointF(result.m_rect.maxX(), result.m_rect.minY())));
|
||||
item->setFlags(Qt::ItemIsSelectable | Qt::ItemIsEnabled);
|
||||
table->setItem(rowCount, 0, item);
|
||||
}
|
||||
}
|
||||
|
||||
void MainWindow::OnSearchPanelShortcutPressed()
|
||||
|
|
|
@ -6,6 +6,11 @@
|
|||
|
||||
class QDockWidget;
|
||||
|
||||
namespace search
|
||||
{
|
||||
class Result;
|
||||
}
|
||||
|
||||
namespace qt
|
||||
{
|
||||
class DrawWidget;
|
||||
|
@ -38,7 +43,7 @@ namespace qt
|
|||
|
||||
private:
|
||||
void OnLocationFound();
|
||||
void OnSearchResult(string const & name, m2::RectD const & rect);
|
||||
void OnSearchResult(search::Result const & result);
|
||||
|
||||
protected:
|
||||
#ifdef DEBUG // code removed for desktop releases
|
||||
|
|
Loading…
Add table
Reference in a new issue