limiting max and min global rect.

This commit is contained in:
rachytski 2011-03-15 01:45:13 +02:00 committed by Alex Zolotarev
parent 90a1c8a23e
commit b8f6f73964
5 changed files with 63 additions and 22 deletions

View file

@ -175,6 +175,8 @@ public:
m_informationDisplay.enableCenter(false);
m_informationDisplay.enableGlobalRect(!m_isBenchmarking);
m_informationDisplay.enableRuler(true);
m_informationDisplay.setRulerParams(80, 20);
m_navigator.SetMinScreenParams(80, 20);
#ifdef DEBUG
m_informationDisplay.enableDebugInfo(true);
@ -378,7 +380,7 @@ public:
m2::PointD const center = m_navigator.Screen().ClipRect().Center();
m_informationDisplay.setGlobalRect(m_navigator.Screen().GlobalRect());
m_informationDisplay.setCenter(m2::PointD(MercatorBounds::YToLat(center.y), MercatorBounds::XToLon(center.x)));
m_informationDisplay.setCenter(m2::PointD(MercatorBounds::XToLon(center.x), MercatorBounds::YToLat(center.y)));
{
threads::MutexGuard guard(*m_renderQueue.renderState().m_mutex.get());
@ -387,7 +389,7 @@ public:
{
m2::PointD const center = m_renderQueue.renderState().m_actualScreen.ClipRect().Center();
m_informationDisplay.setScreen(m_renderQueue.renderState().m_actualScreen);
m_informationDisplay.setCenter(m2::PointD(MercatorBounds::YToLat(center.y), MercatorBounds::XToLon(center.x)));
m_informationDisplay.setCenter(m2::PointD(MercatorBounds::XToLon(center.x), MercatorBounds::YToLat(center.y)));
if (m_benchmarks.empty())
{

View file

@ -171,21 +171,27 @@ void InformationDisplay::enableRuler(bool doEnable)
m_isRulerEnabled = doEnable;
}
void InformationDisplay::setRulerParams(unsigned pxMinWidth, double metresMinWidth)
{
m_pxMinWidth = pxMinWidth;
m_metresMinWidth = metresMinWidth;
}
void InformationDisplay::drawRuler(DrawerYG * pDrawer)
{
/// Compute Scaler
/// scaler should be between minPixSize and maxPixSize
int minPixSize = 80;
int minPixSize = m_pxMinWidth;
m2::PointD pt0 = m_centerPt;
m2::PointD pt1 = m_screen.PtoG(m_screen.GtoP(m_centerPt) + m2::PointD(minPixSize, 0));
double latDiff = fabs(MercatorBounds::YToLat(pt1.x) - MercatorBounds::YToLat(pt0.x));
double metresDiff = latDiff / MercatorBounds::degreeInMetres;
double lonDiff = fabs(MercatorBounds::XToLon(pt1.x) - MercatorBounds::XToLon(pt0.x));
double metresDiff = lonDiff / MercatorBounds::degreeInMetres;
/// finding the closest higher metric value
unsigned curFirstDigit = 2;
unsigned curVal = 20;
unsigned curVal = m_metresMinWidth;
unsigned maxVal = 1000000;
bool lessThanMin = false;
bool isInfinity = false;
@ -220,7 +226,7 @@ void InformationDisplay::drawRuler(DrawerYG * pDrawer)
/// translating meters to pixels
double scalerWidthLatDiff = (double)curVal * MercatorBounds::degreeInMetres;
double scalerWidthXDiff = MercatorBounds::LatToY(pt0.x + scalerWidthLatDiff) - MercatorBounds::LatToY(pt0.x);
double scalerWidthXDiff = MercatorBounds::LonToX(pt0.x + scalerWidthLatDiff) - MercatorBounds::LonToX(pt0.x);
double scalerWidthInPx = m_screen.GtoP(pt0).x - m_screen.GtoP(pt0 + m2::PointD(scalerWidthXDiff, 0)).x;
scalerWidthInPx = (lessThanMin || isInfinity) ? minPixSize : abs(my::rounds(scalerWidthInPx));

View file

@ -37,6 +37,8 @@ private:
bool m_isRulerEnabled;
m2::PointD m_basePoint;
unsigned m_pxMinWidth;
double m_metresMinWidth;
bool m_isCenterEnabled;
m2::PointD m_centerPt;
@ -96,6 +98,7 @@ public:
void enableRuler(bool doEnable);
void drawRuler(DrawerYG * pDrawer);
void setRulerParams(unsigned pxMinWidth, double metresMinWidth);
void enableCenter(bool doEnable);
void setCenter(m2::PointD const & latLongPt);

View file

@ -29,6 +29,12 @@ Navigator::Navigator(ScreenBase const & screen)
{
}
void Navigator::SetMinScreenParams(unsigned pxMinWidth, double metresMinWidth)
{
m_pxMinWidth = pxMinWidth;
m_metresMinWidth = metresMinWidth;
}
void Navigator::SetFromRect(m2::RectD const & r)
{
m_Screen.SetFromRect(r);
@ -148,23 +154,37 @@ void Navigator::ScaleToPoint(m2::PointD const & pt, double factor, double /*time
endPt.y = m_Screen.PixelRect().minY();
}
ScaleImpl(pt, endPt, pt, startPt);
ScaleImpl(pt, endPt, pt, startPt, factor > 1);
}
namespace
bool Navigator::CheckMaxScale(ScreenBase const & screen)
{
bool CheckMaxScale(ScreenBase const & screen)
{
m2::RectD const r = screen.GlobalRect();
m2::RectD const r0 = screen.GlobalRect();
m2::RectD const r1 = screen.ClipRect();
m2::RectD r;
// multiple by 2 to allow scale on zero level
double const maxSize = 2.0 * (MercatorBounds::maxX - MercatorBounds::minX);
return (r.SizeX() <= maxSize || r.SizeY() <= maxSize);
}
if ((r0.SizeX() > r1.SizeX()) || (r0.SizeY() > r1.SizeY()))
r = r0;
else
r = r1;
// multiple by 2 to allow scale on zero level
double const maxSize = (MercatorBounds::maxX - MercatorBounds::minX);
return (r.SizeX() <= maxSize || r.SizeY() <= maxSize);
}
bool Navigator::CheckMinScale(ScreenBase const & screen)
{
m2::PointD const pt0 = screen.GlobalRect().Center();
m2::PointD const pt1 = screen.PtoG(screen.GtoP(pt0) + m2::PointD(m_pxMinWidth, 0));
double lonDiff = fabs(MercatorBounds::XToLon(pt1.x) - MercatorBounds::XToLon(pt0.x));
double metresDiff = lonDiff / MercatorBounds::degreeInMetres;
return metresDiff >= m_metresMinWidth - 1;
}
void Navigator::ScaleImpl(m2::PointD const & newPt1, m2::PointD const & newPt2,
m2::PointD const & oldPt1, m2::PointD const & oldPt2)
m2::PointD const & oldPt1, m2::PointD const & oldPt2,
bool skipMaxScaleCheck)
{
math::Matrix<double, 3, 3> newM = m_Screen.GtoPMatrix() * ScreenBase::CalcTransform(oldPt1, oldPt2, newPt1, newPt2);
@ -172,8 +192,10 @@ void Navigator::ScaleImpl(m2::PointD const & newPt1, m2::PointD const & newPt2,
tmp.SetGtoPMatrix(newM);
tmp.Rotate(tmp.GetAngle());
// limit max scale to MercatorBounds
if (CheckMaxScale(tmp))
if ((!skipMaxScaleCheck) && (!CheckMaxScale(tmp)))
return;
if (CheckMinScale(tmp))
m_Screen = tmp;
}
@ -186,7 +208,7 @@ void Navigator::DoScale(m2::PointD const & pt1, m2::PointD const & pt2, double /
return;
m_Screen = m_StartScreen;
ScaleImpl(pt1, pt2, m_StartPt1, m_StartPt2);
ScaleImpl(pt1, pt2, m_StartPt1, m_StartPt2, pt1.Length(pt2) / m_StartPt1.Length(m_StartPt2) > 1);
m_LastPt1 = pt1;
m_LastPt2 = pt2;
@ -209,7 +231,7 @@ void Navigator::Scale(double scale)
tmp.Scale(scale);
// limit max scale to MercatorBounds
if (CheckMaxScale(tmp))
if (CheckMaxScale(tmp) && CheckMinScale(tmp))
m_Screen = tmp;
}

View file

@ -17,6 +17,7 @@ public:
Navigator();
explicit Navigator(ScreenBase const & screen);
void SetMinScreenParams(unsigned pxMinWidth, double metresMinWidth);
void SetFromRect(m2::RectD const & r);
void CenterViewport(m2::PointD const & p);
@ -60,6 +61,13 @@ public:
private:
unsigned m_pxMinWidth;
double m_metresMinWidth;
bool CheckMinScale(ScreenBase const & screen);
bool CheckMaxScale(ScreenBase const & screen);
// Internal screen corresponding to the state when navigation began with StartDrag or StartScale.
ScreenBase m_StartScreen;
// Internal screen to do GtoP() and PtoG() calculations. It is always up to date with navigation.
@ -85,7 +93,7 @@ private:
// Device orientation
EOrientation m_orientation;
// Used in DoScale and ScaleByPoint
void ScaleImpl(m2::PointD const & newPt1, m2::PointD const & newPt2, m2::PointD const & oldPt1, m2::PointD const & oldPt2);
void ScaleImpl(m2::PointD const & newPt1, m2::PointD const & newPt2, m2::PointD const & oldPt1, m2::PointD const & oldPt2, bool skipMaxScaleCheck);
};
#include "../base/stop_mem_debug.hpp"