Show types context menu for right button click on features.

This commit is contained in:
vng 2011-12-02 13:03:20 +03:00 committed by Alex Zolotarev
parent 7e8bd03296
commit 44d584eb5e
5 changed files with 222 additions and 12 deletions

View file

@ -2,6 +2,9 @@
#include "feature_visibility.hpp"
#include "feature_loader_base.hpp"
#include "../geometry/distance.hpp"
#include "../geometry/robust_orientation.hpp"
#include "../platform/preferred_languages.hpp"
#include "../defines.hpp" // just for file extensions
@ -290,3 +293,81 @@ double FeatureType::GetPopulationDrawRank() const
return min(upperBound, static_cast<double>(n)) / upperBound;
}
}
namespace
{
class DoCalcDistance
{
m2::PointD m_prev, m_pt;
bool m_hasPrev;
static double Inf() { return numeric_limits<double>::max(); }
static double GetDistance(m2::PointD const & p1, m2::PointD const & p2, m2::PointD const & p)
{
mn::DistanceToLineSquare<m2::PointD> calc;
calc.SetBounds(p1, p2);
return sqrt(calc(p));
}
public:
DoCalcDistance(m2::PointD const & pt)
: m_pt(pt), m_hasPrev(false), m_dist(Inf())
{
}
void TestPoint(m2::PointD const & p)
{
m_dist = m_pt.Length(p);
}
void operator() (CoordPointT const & p)
{
m2::PointD pt(p.first, p.second);
if (m_hasPrev)
m_dist = min(m_dist, GetDistance(m_prev, pt, m_pt));
else
m_hasPrev = true;
m_prev = pt;
}
void operator() (m2::PointD const & p1, m2::PointD const & p2, m2::PointD const & p3)
{
m2::PointD arrP[] = { p1, p2, p3 };
// make right-oriented triangle
if (m2::robust::OrientedS(arrP[0], arrP[1], arrP[2]) < 0.0)
swap(arrP[1], arrP[2]);
double d = Inf();
for (size_t i = 0; i < 3; ++i)
{
double const s = m2::robust::OrientedS(arrP[i], arrP[(i + 1) % 3], m_pt);
if (s < 0.0)
d = min(d, GetDistance(arrP[i], arrP[(i + 1) % 3], m_pt));
}
m_dist = ((d == Inf()) ? 0.0 : min(m_dist, d));
}
double m_dist;
};
}
double FeatureType::GetDistance(m2::PointD const & pt, int scale) const
{
DoCalcDistance calc(pt);
switch (GetFeatureType())
{
case GEOM_POINT: calc.TestPoint(GetCenter()); break;
case GEOM_LINE: ForEachPointRef(calc, scale); break;
case GEOM_AREA: ForEachTriangleRef(calc, scale); break;
default:
CHECK ( false, () );
}
return calc.m_dist;
}

View file

@ -239,6 +239,8 @@ public:
inline string GetRoadNumber() const { return m_Params.ref; }
double GetDistance(m2::PointD const & pt, int scale) const;
/// @name Statistic functions.
//@{
inline void ParseBeforeStatistic() const

View file

@ -13,19 +13,16 @@
#include "../search/result.hpp"
#include "../indexer/categories_holder.hpp"
#include "../indexer/feature_visibility.hpp"
#include "../indexer/feature.hpp"
#include "../indexer/scales.hpp"
#include "../indexer/drawing_rules.hpp"
#include "../base/math.hpp"
#include "../base/string_utils.hpp"
#include "../std/algorithm.hpp"
#include "../std/fstream.hpp"
#include "../std/target_os.hpp"
#include "../std/vector.hpp"
using namespace feature;
void Framework::AddMap(string const & file)
{
@ -250,10 +247,10 @@ RenderPolicy::TRenderFn Framework::DrawModelFn()
/// Actual rendering function.
void Framework::DrawModel(shared_ptr<PaintEvent> const & e,
ScreenBase const & screen,
m2::RectD const & selectRect,
m2::RectD const & clipRect,
int scaleLevel)
ScreenBase const & screen,
m2::RectD const & selectRect,
m2::RectD const & clipRect,
int scaleLevel)
{
fwork::DrawProcessor doDraw(clipRect, screen, e, scaleLevel);
@ -676,3 +673,112 @@ void Framework::DeleteOldMaps()
}
m_lowestMapVersion = MAXIMUM_VERSION_TO_DELETE + 1;
}
namespace
{
class DoGetFeatureTypes
{
typedef vector<string> TypesC;
class DistanceT
{
double m_dist;
public:
DistanceT(double d, TypesC & types) : m_dist(d)
{
m_types.swap(types);
}
bool operator<(DistanceT const & rhs) const
{
return (m_dist < rhs.m_dist);
}
TypesC m_types;
};
class DoParseTypes
{
Classificator const & m_c;
public:
DoParseTypes() : m_c(classif()) {}
void operator() (uint32_t t)
{
m_types.push_back(m_c.GetFullObjectName(t));
}
TypesC m_types;
};
double GetCompareEpsilon(feature::EGeomType type) const
{
using namespace feature;
switch (type)
{
case GEOM_POINT: return 0.0 * m_eps;
case GEOM_LINE: return 1.0 * m_eps;
case GEOM_AREA: return 2.0 * m_eps;
default:
ASSERT ( false, () );
return numeric_limits<double>::max();
}
}
m2::PointD m_pt;
double m_eps;
int m_scale;
vector<DistanceT> m_cont;
public:
DoGetFeatureTypes(m2::PointD const & pt, double eps, int scale)
: m_pt(pt), m_eps(eps), m_scale(scale)
{
}
void operator() (FeatureType const & f)
{
double const d = f.GetDistance(m_pt, m_scale);
ASSERT_GREATER_OR_EQUAL(d, 0.0, ());
if (d <= m_eps)
{
DoParseTypes doParse;
f.ForEachTypeRef(doParse);
m_cont.push_back(DistanceT(d + GetCompareEpsilon(f.GetFeatureType()), doParse.m_types));
}
}
void GetFeatureTypes(size_t count, TypesC & types)
{
sort(m_cont.begin(), m_cont.end());
for (size_t i = 0; i < min(count, m_cont.size()); ++i)
types.insert(types.end(), m_cont[i].m_types.begin(), m_cont[i].m_types.end());
}
};
}
void Framework::GetFeatureTypes(m2::PointD pt, vector<string> & types) const
{
pt = m_navigator.ShiftPoint(pt);
int const sm = 20;
m2::RectD pixR(m2::PointD(pt.x - sm, pt.y - sm), m2::PointD(pt.x + sm, pt.y + sm));
m2::RectD glbR;
m_navigator.Screen().PtoG(pixR, glbR);
int const scale = my::rounds(GetCurrentScale());
DoGetFeatureTypes getTypes(m_navigator.Screen().PtoG(pt),
max(glbR.SizeX() / 2.0, glbR.SizeY() / 2.0),
scale);
m_model.ForEachFeature(glbR, getTypes, scale);
getTypes.GetFeatureTypes(5, types);
}

View file

@ -172,6 +172,8 @@ public:
Invalidate(true);
}
void GetFeatureTypes(m2::PointD pt, vector<string> & types) const;
virtual void BeginPaint(shared_ptr<PaintEvent> const & e);
/// Function for calling from platform dependent-paint function.
virtual void DoPaint(shared_ptr<PaintEvent> const & e);

View file

@ -10,6 +10,7 @@
#include "../platform/settings.hpp"
#include <QtGui/QMouseEvent>
#include <QtGui/QMenu>
using namespace storage;
@ -287,20 +288,35 @@ namespace qt
{
if (e->modifiers() & Qt::ControlModifier)
{
/// starting rotation
// starting rotation
m_framework->StartRotate(get_rotate_event(e->pos(), this->rect().center()));
setCursor(Qt::CrossCursor);
m_isRotate = true;
}
else
{
/// starting drag
// starting drag
m_framework->StartDrag(get_drag_event(e));
setCursor(Qt::CrossCursor);
m_isDrag = true;
}
}
else if (e->button() == Qt::RightButton)
{
// show feature types
QPoint const & pt = e->pos();
vector<string> types;
m_framework->GetFeatureTypes(m2::PointD(pt.x(), pt.y()), types);
QMenu menu;
for (size_t i = 0; i < types.size(); ++i)
menu.addAction(QString::fromAscii(types[i].c_str()));
menu.exec(pt);
}
}
void DrawWidget::mouseDoubleClickEvent(QMouseEvent * e)
@ -353,6 +369,7 @@ namespace qt
if (m_isRotate && (e->button() == Qt::LeftButton))
{
m_framework->StopRotate(get_rotate_event(e->pos(), this->rect().center()));
setCursor(Qt::ArrowCursor);
m_isRotate = false;
}
@ -389,15 +406,17 @@ namespace qt
void DrawWidget::wheelEvent(QWheelEvent * e)
{
if ((!m_isDrag) && (!m_isRotate))
if (!m_isDrag && !m_isRotate)
{
/// if we are inside the timer, cancel it
// if we are inside the timer, cancel it
if (m_timer->isActive())
m_timer->stop();
m_timer->start(m_redrawInterval);
//m_framework->Scale(exp(e->delta() / 360.0));
m_framework->ScaleToPoint(ScaleToPointEvent(e->pos().x(), e->pos().y(), exp(e->delta() / 360.0)));
UpdateScaleControl();
emit ViewportChanged();
}