City boundaries visualization.
This commit is contained in:
parent
b5e49972d6
commit
f79ed2af2c
10 changed files with 157 additions and 12 deletions
|
@ -20,6 +20,7 @@
|
|||
|
||||
#include "routing_common/num_mwm_id.hpp"
|
||||
|
||||
#include "search/cities_boundaries_table.hpp"
|
||||
#include "search/downloader_search_callback.hpp"
|
||||
#include "search/editor_delegate.hpp"
|
||||
#include "search/engine.hpp"
|
||||
|
@ -3119,12 +3120,34 @@ storage::TCountriesVec Framework::GetTopmostCountries(ms::LatLon const & latlon)
|
|||
|
||||
namespace
|
||||
{
|
||||
vector<dp::Color> colorList = { dp::Color(255, 0, 0, 255), dp::Color(0, 255, 0, 255), dp::Color(0, 0, 255, 255),
|
||||
dp::Color(255, 255, 0, 255), dp::Color(0, 255, 255, 255), dp::Color(255, 0, 255, 255),
|
||||
dp::Color(100, 0, 0, 255), dp::Color(0, 100, 0, 255), dp::Color(0, 0, 100, 255),
|
||||
dp::Color(100, 100, 0, 255), dp::Color(0, 100, 100, 255), dp::Color(100, 0, 100, 255)
|
||||
};
|
||||
} // namespace
|
||||
vector<dp::Color> colorList = {
|
||||
dp::Color(255, 0, 0, 255), dp::Color(0, 255, 0, 255), dp::Color(0, 0, 255, 255),
|
||||
dp::Color(255, 255, 0, 255), dp::Color(0, 255, 255, 255), dp::Color(255, 0, 255, 255),
|
||||
dp::Color(100, 0, 0, 255), dp::Color(0, 100, 0, 255), dp::Color(0, 0, 100, 255),
|
||||
dp::Color(100, 100, 0, 255), dp::Color(0, 100, 100, 255), dp::Color(100, 0, 100, 255)};
|
||||
|
||||
dp::Color const cityBoundaryBBColor = dp::Color(255, 0, 0, 255);
|
||||
dp::Color const cityBoundaryCBColor = dp::Color(0, 255, 0, 255);
|
||||
dp::Color const cityBoundaryDBColor = dp::Color(0, 0, 255, 255);
|
||||
|
||||
template <class Box>
|
||||
void DrawLine(Box const & box, dp::Color const & color, df::DrapeApi & drapeApi, string const & id)
|
||||
{
|
||||
auto points = box.Points();
|
||||
CHECK(!points.empty(), ());
|
||||
points.push_back(points.front());
|
||||
|
||||
points.erase(unique(points.begin(), points.end(), [](m2::PointD const & p1, m2::PointD const & p2) {
|
||||
m2::PointD const delta = p2 - p1;
|
||||
return delta.IsAlmostZero();
|
||||
}), points.end());
|
||||
|
||||
if (points.size() <= 1)
|
||||
return;
|
||||
|
||||
drapeApi.AddLine(id, df::DrapeApiLineData(points, color).Width(3.0f).ShowPoints(true).ShowId());
|
||||
}
|
||||
} // namespace
|
||||
|
||||
void Framework::VisualizeRoadsInRect(m2::RectD const & rect)
|
||||
{
|
||||
|
@ -3155,6 +3178,44 @@ void Framework::VisualizeRoadsInRect(m2::RectD const & rect)
|
|||
}, kScale);
|
||||
}
|
||||
|
||||
void Framework::VisualizeCityBoundariesInRect(m2::RectD const & rect)
|
||||
{
|
||||
search::CitiesBoundariesTable table(GetDataSource());
|
||||
table.Load();
|
||||
|
||||
vector<uint32_t> featureIds;
|
||||
GetCityBoundariesInRectForTesting(table, rect, featureIds);
|
||||
|
||||
for (auto const fid : featureIds)
|
||||
{
|
||||
search::CitiesBoundariesTable::Boundaries boundaries;
|
||||
table.Get(fid, boundaries);
|
||||
|
||||
string id = "fid:" + strings::to_string(fid);
|
||||
FeaturesLoaderGuard loader(GetDataSource(), GetDataSource().GetMwmIdByCountryFile(CountryFile("World")));
|
||||
FeatureType ft;
|
||||
if (loader.GetFeatureByIndex(fid, ft))
|
||||
{
|
||||
string name;
|
||||
ft.GetName(FeatureType::DEFAULT_LANG, name);
|
||||
id += ", name:" + name;
|
||||
}
|
||||
|
||||
size_t const boundariesSize = boundaries.GetBoundariesForTesting().size();
|
||||
for (size_t i = 0; i < boundariesSize; ++i)
|
||||
{
|
||||
string idWithIndex = id;
|
||||
auto const & cityBoundary = boundaries.GetBoundariesForTesting()[i];
|
||||
if (boundariesSize > 1)
|
||||
idWithIndex = id + " , i:" + strings::to_string(i);
|
||||
|
||||
DrawLine(cityBoundary.m_bbox, cityBoundaryBBColor, m_drapeApi, idWithIndex + ", bb");
|
||||
DrawLine(cityBoundary.m_cbox, cityBoundaryCBColor, m_drapeApi, idWithIndex + ", cb");
|
||||
DrawLine(cityBoundary.m_dbox, cityBoundaryDBColor, m_drapeApi, idWithIndex + ", db");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
ads::Engine const & Framework::GetAdsEngine() const
|
||||
{
|
||||
ASSERT(m_adsEngine, ());
|
||||
|
|
|
@ -339,6 +339,7 @@ public:
|
|||
|
||||
// Utilities
|
||||
void VisualizeRoadsInRect(m2::RectD const & rect);
|
||||
void VisualizeCityBoundariesInRect(m2::RectD const & rect);
|
||||
|
||||
ads::Engine const & GetAdsEngine() const;
|
||||
|
||||
|
|
|
@ -19,6 +19,8 @@
|
|||
#include "platform/platform.hpp"
|
||||
#include "platform/settings.hpp"
|
||||
|
||||
#include "base/assert.hpp"
|
||||
|
||||
#include <QtGui/QMouseEvent>
|
||||
#include <QtGui/QGuiApplication>
|
||||
|
||||
|
@ -187,7 +189,7 @@ void DrawWidget::mousePressEvent(QMouseEvent * e)
|
|||
{
|
||||
SubmitBookmark(pt);
|
||||
}
|
||||
else if (!m_selectionMode || IsCommandModifier(e))
|
||||
else if (!(m_selectionMode || m_cityBoundariesSelectionMode) || IsCommandModifier(e))
|
||||
{
|
||||
ShowInfoPopup(e, pt);
|
||||
}
|
||||
|
@ -208,7 +210,8 @@ void DrawWidget::mouseMoveEvent(QMouseEvent * e)
|
|||
if (IsLeftButton(e) && !IsAltModifier(e))
|
||||
m_framework.TouchEvent(GetTouchEvent(e, df::TouchEvent::TOUCH_MOVE));
|
||||
|
||||
if (m_selectionMode && m_rubberBand != nullptr && m_rubberBand->isVisible())
|
||||
if ((m_selectionMode || m_cityBoundariesSelectionMode) && m_rubberBand != nullptr &&
|
||||
m_rubberBand->isVisible())
|
||||
{
|
||||
m_rubberBand->setGeometry(QRect(m_rubberBandOrigin, e->pos()).normalized());
|
||||
}
|
||||
|
@ -221,15 +224,25 @@ void DrawWidget::mouseReleaseEvent(QMouseEvent * e)
|
|||
{
|
||||
m_framework.TouchEvent(GetTouchEvent(e, df::TouchEvent::TOUCH_UP));
|
||||
}
|
||||
else if (m_selectionMode && IsRightButton(e) && m_rubberBand != nullptr &&
|
||||
m_rubberBand->isVisible())
|
||||
else if ((m_selectionMode || m_cityBoundariesSelectionMode) && IsRightButton(e) &&
|
||||
m_rubberBand != nullptr && m_rubberBand->isVisible())
|
||||
{
|
||||
QPoint const lt = m_rubberBand->geometry().topLeft();
|
||||
QPoint const rb = m_rubberBand->geometry().bottomRight();
|
||||
m2::RectD rect;
|
||||
rect.Add(m_framework.PtoG(m2::PointD(L2D(lt.x()), L2D(lt.y()))));
|
||||
rect.Add(m_framework.PtoG(m2::PointD(L2D(rb.x()), L2D(rb.y()))));
|
||||
m_framework.VisualizeRoadsInRect(rect);
|
||||
if (m_selectionMode)
|
||||
{
|
||||
CHECK(!m_cityBoundariesSelectionMode, ());
|
||||
m_framework.VisualizeRoadsInRect(rect);
|
||||
}
|
||||
else
|
||||
{
|
||||
CHECK(m_cityBoundariesSelectionMode, ());
|
||||
m_framework.VisualizeCityBoundariesInRect(rect);
|
||||
}
|
||||
|
||||
m_rubberBand->hide();
|
||||
}
|
||||
}
|
||||
|
@ -507,6 +520,11 @@ void DrawWidget::SetRouter(routing::RouterType routerType)
|
|||
|
||||
void DrawWidget::SetSelectionMode(bool mode) { m_selectionMode = mode; }
|
||||
|
||||
void DrawWidget::SetCityBoundariesSelectionMode(bool mode)
|
||||
{
|
||||
m_cityBoundariesSelectionMode = mode;
|
||||
}
|
||||
|
||||
// static
|
||||
void DrawWidget::SetDefaultSurfaceFormat(bool apiOpenGLES3)
|
||||
{
|
||||
|
|
|
@ -73,6 +73,7 @@ public:
|
|||
void RetryToDownloadCountry(storage::TCountryId const & countryId);
|
||||
|
||||
void SetSelectionMode(bool mode);
|
||||
void SetCityBoundariesSelectionMode(bool mode);
|
||||
|
||||
RouteMarkType GetRoutePointAddMode() const { return m_routePointAddMode; }
|
||||
void SetRoutePointAddMode(RouteMarkType mode) { m_routePointAddMode = mode; }
|
||||
|
@ -114,6 +115,7 @@ private:
|
|||
storage::TCountryId m_countryId;
|
||||
|
||||
bool m_selectionMode = false;
|
||||
bool m_cityBoundariesSelectionMode = false;
|
||||
RouteMarkType m_routePointAddMode = RouteMarkType::Finish;
|
||||
};
|
||||
} // namespace qt
|
||||
|
|
|
@ -323,6 +323,12 @@ void MainWindow::CreateNavigationBar()
|
|||
m_selectionMode->setCheckable(true);
|
||||
m_selectionMode->setToolTip(tr("Turn on/off selection mode"));
|
||||
|
||||
m_selectionCityBoundariesMode =
|
||||
pToolBar->addAction(QIcon(":/navig64/city_boundaries.png"), tr("City boundaries selection mode"),
|
||||
this, SLOT(OnSwitchCityBoundariesSelectionMode()));
|
||||
m_selectionCityBoundariesMode->setCheckable(true);
|
||||
m_selectionCityBoundariesMode->setChecked(m_pDrawWidget->GetFramework().LoadTrafficEnabled());
|
||||
|
||||
m_clearSelection = pToolBar->addAction(QIcon(":/navig64/clear.png"), tr("Clear selection"),
|
||||
this, SLOT(OnClearSelection()));
|
||||
m_clearSelection->setToolTip(tr("Clear selection"));
|
||||
|
@ -549,10 +555,32 @@ void MainWindow::OnCreateFeatureClicked()
|
|||
|
||||
void MainWindow::OnSwitchSelectionMode()
|
||||
{
|
||||
if (m_selectionCityBoundariesMode->isChecked())
|
||||
{
|
||||
// Unchecking selection city boundaries mode.
|
||||
m_selectionCityBoundariesMode->setChecked(false);
|
||||
m_pDrawWidget->SetCityBoundariesSelectionMode(false);
|
||||
}
|
||||
|
||||
m_pDrawWidget->SetSelectionMode(m_selectionMode->isChecked());
|
||||
}
|
||||
|
||||
void MainWindow::OnClearSelection() { m_pDrawWidget->GetFramework().GetDrapeApi().Clear(); }
|
||||
void MainWindow::OnSwitchCityBoundariesSelectionMode()
|
||||
{
|
||||
if (m_selectionMode->isChecked())
|
||||
{
|
||||
// Unchecking selection mode.
|
||||
m_selectionMode->setChecked(false);
|
||||
m_pDrawWidget->SetSelectionMode(false);
|
||||
}
|
||||
|
||||
m_pDrawWidget->SetCityBoundariesSelectionMode(m_selectionCityBoundariesMode->isChecked());
|
||||
}
|
||||
|
||||
void MainWindow::OnClearSelection()
|
||||
{
|
||||
m_pDrawWidget->GetFramework().GetDrapeApi().Clear();
|
||||
}
|
||||
|
||||
void MainWindow::OnSearchButtonClicked()
|
||||
{
|
||||
|
|
|
@ -45,6 +45,7 @@ class MainWindow : public QMainWindow, location::LocationObserver
|
|||
QAction * m_clearSelection = nullptr;
|
||||
QAction * m_pSearchAction = nullptr;
|
||||
QAction * m_trafficEnableAction = nullptr;
|
||||
QAction * m_selectionCityBoundariesMode = nullptr;
|
||||
QToolButton * m_routePointsToolButton = nullptr;
|
||||
QAction * m_selectStartRoutePoint = nullptr;
|
||||
QAction * m_selectFinishRoutePoint = nullptr;
|
||||
|
@ -105,6 +106,7 @@ protected Q_SLOTS:
|
|||
void OnRetryDownloadClicked();
|
||||
|
||||
void OnSwitchSelectionMode();
|
||||
void OnSwitchCityBoundariesSelectionMode();
|
||||
void OnClearSelection();
|
||||
|
||||
void OnTrafficEnabled();
|
||||
|
|
BIN
qt/res/city_boundaries.png
Normal file
BIN
qt/res/city_boundaries.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 904 B |
|
@ -13,6 +13,7 @@
|
|||
<file>selectmode.png</file>
|
||||
<file>clear.png</file>
|
||||
<file>traffic.png</file>
|
||||
<file>city_boundaries.png</file>
|
||||
<file>point-finish.png</file>
|
||||
<file>point-intermediate.png</file>
|
||||
<file>point-start.png</file>
|
||||
|
|
|
@ -105,4 +105,21 @@ bool CitiesBoundariesTable::Get(uint32_t fid, Boundaries & bs) const
|
|||
bs = Boundaries(it->second, m_eps);
|
||||
return true;
|
||||
}
|
||||
|
||||
void GetCityBoundariesInRectForTesting(CitiesBoundariesTable const & table, m2::RectD const & rect,
|
||||
vector<uint32_t> & featureIds)
|
||||
{
|
||||
featureIds.clear();
|
||||
for (auto const & kv : table.m_table)
|
||||
{
|
||||
for (auto const & cb : kv.second)
|
||||
{
|
||||
if (rect.IsIntersect(m2::RectD(cb.m_bbox.Min(), cb.m_bbox.Max())))
|
||||
{
|
||||
featureIds.push_back(kv.first);
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
} // namespace search
|
||||
|
|
|
@ -4,6 +4,7 @@
|
|||
#include "indexer/feature_decl.hpp"
|
||||
|
||||
#include "geometry/point2d.hpp"
|
||||
#include "geometry/rect2d.hpp"
|
||||
|
||||
#include <cstdint>
|
||||
#include <sstream>
|
||||
|
@ -23,6 +24,10 @@ namespace search
|
|||
{
|
||||
class CitiesBoundariesTable
|
||||
{
|
||||
friend void GetCityBoundariesInRectForTesting(CitiesBoundariesTable const &,
|
||||
m2::RectD const & rect,
|
||||
std::vector<uint32_t> & featureIds);
|
||||
|
||||
public:
|
||||
class Boundaries
|
||||
{
|
||||
|
@ -43,6 +48,8 @@ public:
|
|||
// |*this|.
|
||||
bool HasPoint(m2::PointD const & p) const;
|
||||
|
||||
std::vector<indexer::CityBoundary> const & GetBoundariesForTesting() const { return m_boundaries; }
|
||||
|
||||
friend std::string DebugPrint(Boundaries const & boundaries)
|
||||
{
|
||||
std::ostringstream os;
|
||||
|
@ -68,10 +75,18 @@ public:
|
|||
bool Get(FeatureID const & fid, Boundaries & bs) const;
|
||||
bool Get(uint32_t fid, Boundaries & bs) const;
|
||||
|
||||
size_t GetSize() const { return m_table.size(); }
|
||||
|
||||
private:
|
||||
DataSource const & m_dataSource;
|
||||
MwmSet::MwmId m_mwmId;
|
||||
std::unordered_map<uint32_t, std::vector<indexer::CityBoundary>> m_table;
|
||||
double m_eps = 0.0;
|
||||
};
|
||||
|
||||
/// \brief Fills |featureIds| with feature ids of city boundaries if bounding rect of
|
||||
/// the city boundary crosses |rect|.
|
||||
/// \note This method is inefficient and is written for debug and test purposes only.
|
||||
void GetCityBoundariesInRectForTesting(CitiesBoundariesTable const &, m2::RectD const & rect,
|
||||
std::vector<uint32_t> & featureIds);
|
||||
} // namespace search
|
||||
|
|
Reference in a new issue