mathplot: further improvements for simulator [wip]

This commit is contained in:
Tomasz Wlostowski 2016-08-11 14:41:49 +02:00 committed by Maciej Suminski
parent 30c87527da
commit b9e31f6d3a
2 changed files with 174 additions and 34 deletions

View File

@ -553,6 +553,7 @@ void mpFXY::Plot(wxDC & dc, mpWindow & w)
if (m_visible) {
dc.SetPen( m_pen);
double x, y;
// Do this to reset the counters to evaluate bounding box for label positioning
Rewind(); GetNextXY(x, y);
@ -565,6 +566,8 @@ void mpFXY::Plot(wxDC & dc, mpWindow & w)
wxCoord minYpx = m_drawOutsideMargins ? 0 : w.GetMarginTop();
wxCoord maxYpx = m_drawOutsideMargins ? w.GetScrY() : w.GetScrY() - w.GetMarginBottom();
dc.SetClippingRegion (startPx, minYpx, endPx - startPx + 1, maxYpx - minYpx + 1);
wxCoord ix = 0, iy = 0;
if (!m_continuous)
@ -607,6 +610,7 @@ void mpFXY::Plot(wxDC & dc, mpWindow & w)
}
else
{
int n =0 ;
// Old code
wxCoord x0=0,c0=0;
bool first = TRUE;
@ -615,6 +619,9 @@ void mpFXY::Plot(wxDC & dc, mpWindow & w)
double px = m_scaleX->TransformToPlot(x);
double py = m_scaleY->TransformToPlot(y);
if(py >= 0.0 && py <= 1.0) n++;
//printf("py %.1f\n", py);
wxCoord x1 = w.x2p(px);
wxCoord c1 = w.y2p(py);
@ -634,11 +641,12 @@ void mpFXY::Plot(wxDC & dc, mpWindow & w)
outUp = (c0 < minYpx) && (c1 < minYpx);
if ( !outUp && !outDown ) {
dc.DrawLine(x0, c0, x1, c1);
UpdateViewBoundary(x1, c1);
// UpdateViewBoundary(x1, c1);
}
}
x0=x1; c0=c1;
}
//printf("n %s %d\n", (const char *) m_name.c_str(), n);
}
if (!m_name.IsEmpty() && m_showName)
@ -678,6 +686,8 @@ void mpFXY::Plot(wxDC & dc, mpWindow & w)
dc.DrawText( m_name, tx, ty);
}
}
dc.DestroyClippingRegion();
}
//-----------------------------------------------------------------------------
@ -741,17 +751,71 @@ void mpProfile::Plot(wxDC & dc, mpWindow & w)
void mpScaleX::recalculateTicks ( wxDC & dc, mpWindow & w )
{
const int extend = w.GetScrX(); // /2;
double minV, maxV, minVvis, maxVvis;
GetDataRange ( minV, maxV );
getVisibleDataRange ( w, minVvis, maxVvis );
dig = floor( log( 128.0 / w.GetScaleX() ) / mpLN10 );
step = exp( mpLN10 * dig);
end = w.GetPosX() + (double)extend / w.GetScaleX();
m_absVisibleMaxV = std::max(std::abs(minVvis), std::abs(maxVvis));
n0 = floor( (w.GetPosX() ) / step ) * step ;
//printf("%.10f %.10f %.10f", n0, step, end);
//printf("minV %.10f maxV %.10f %.10f %.10f\n", minV, maxV, minVvis, maxVvis);
m_tickValues.clear();
m_labeledTicks.clear();
double minErr = 1000000000000.0;
double bestStep;
int bestCount;
for(int i = 10; i <= 20; i+=2)
{
double step = fabs(maxVvis - minVvis) / (double) i;
double base = pow(10, floor(log10(step)));
//printf("base %.3f\n", base);
double stepInt = floor(step / base) * base;
double err = fabs(step - stepInt);
if(err< minErr)
{
minErr = err;
bestStep = stepInt;
bestCount = i;
}
//printf("step %d %.3f %.3f best %.3f\n",i, step, stepInt, bestStep);
}
double v = floor(minVvis / bestStep) * bestStep;
double zeroOffset = 100000000.0;
//printf("maxVVis %.3f\n", maxVvis);
while ( v < maxVvis )
{
m_tickValues.push_back(v);
if (fabs(v) < zeroOffset)
zeroOffset = fabs(v);
//printf("tick %.3f\n", v);
v+=bestStep;
}
if ( zeroOffset <= bestStep )
{
for( double& t: m_tickValues )
t -= zeroOffset;
}
for( double t: m_tickValues )
m_labeledTicks.push_back(t);
//n0 = floor(minVvis / bestStep) * bestStep;
//end = n0 +
//n0 = floor( (w.GetPosX() ) / step ) * step ;
//printf("zeroOffset:%.3f tickjs : %d\n", zeroOffset, m_tickValues.size());
computeLabelExtents( dc, w );
labelStep = ceil(((double) m_maxLabelWidth + mpMIN_X_AXIS_LABEL_SEPARATION)/(w.GetScaleX()*step))*step;
}
void mpScaleX::computeLabelExtents ( wxDC & dc, mpWindow & w )
@ -771,12 +835,12 @@ void mpScaleX::computeLabelExtents ( wxDC & dc, mpWindow & w )
int mpScaleX::tickCount() const
{
return (int) floor( ( end - n0 ) / step );
return m_tickValues.size();
}
int mpScaleX::labelCount() const
{
return (int) floor( ( end - n0 ) / labelStep );
return m_labeledTicks.size();
}
const wxString mpScaleX::getLabel( int n )
@ -786,12 +850,12 @@ const wxString mpScaleX::getLabel( int n )
double mpScaleX::getTickPos( int n )
{
return n0 + (double) n * step;
return m_tickValues[n];
}
double mpScaleX::getLabelPos( int n )
{
return n0 + (double) n * labelStep;
return m_labeledTicks[n];
}
void mpScaleY::getVisibleDataRange ( mpWindow& w, double &minV, double& maxV)
@ -818,11 +882,18 @@ void mpScaleY::computeSlaveTicks( mpWindow& w )
double alpha, beta;
//printf("ComputeSlaveTicks!\n");
//for(alpha = 1.0; alpha < 1.03; alpha += 0.001)
// for(beta = -0.05; beta < 0.05; beta += 0.001)
{
// for ( double t: m_masterScale->m_tickValues )
// {
if( m_masterScale->m_tickValues.size() == 0)
return;
// printf("NTicks %d\n", m_masterScale->m_tickValues.size());
double p0 = m_masterScale->TransformToPlot(m_masterScale->m_tickValues[0]);
double p1 = m_masterScale->TransformToPlot(m_masterScale->m_tickValues[1]);
@ -833,7 +904,10 @@ void mpScaleY::computeSlaveTicks( mpWindow& w )
double y_slave1 = p1 / m_scale;
double dy_slave = (y_slave1 - y_slave0);
double dy_scaled = ceil( 2.0*(y_slave1 - y_slave0)) / 2.0;
double exponent = floor(log10(dy_slave));
double base = dy_slave/ pow(10.0, exponent);
double dy_scaled = ceil( 2.0* base ) / 2.0 * pow (10.0, exponent );
double minvv, maxvv;
@ -844,7 +918,7 @@ void mpScaleY::computeSlaveTicks( mpWindow& w )
m_scale = 1.0 / ( m_maxV - m_minV );
m_scale *= dy_slave / dy_scaled;
printf("dy %.1f %.1f minV %.1f maxV %.1f\n", dy_slave, dy_scaled, m_minV, m_maxV);
// printf("dy %.10f %.10f minV %.1f maxV %.1f\n", dy_slave, dy_scaled, m_minV, m_maxV);
//minvv = (p0 / m_scale - m_offset);
@ -854,7 +928,7 @@ void mpScaleY::computeSlaveTicks( mpWindow& w )
//m_offset =
//double y0_offset = floor ( (p0 / m_scale ) / dy_scaled) * dy_scaled;
printf("P0 %.10f minvv %.10f\n", (p0 / m_scale) - m_offset, minvv);
//printf("P0 %.10f minvv %.10f\n", (p0 / m_scale) - m_offset, minvv);
m_tickValues.clear();
@ -865,7 +939,7 @@ void mpScaleY::computeSlaveTicks( mpWindow& w )
for (int i = 0; i < m_masterScale->m_tickValues.size(); i++)
{
m = TransformFromPlot ( m_masterScale->TransformToPlot(m_masterScale->m_tickValues[i]) );
printf("m %.10f\n", m);
// printf("m %.10f\n", m);
m_tickValues.push_back(m);
m_labeledTicks.push_back(m);
//m += dy_scaled;
@ -881,8 +955,8 @@ void mpScaleY::computeSlaveTicks( mpWindow& w )
void mpScaleY::recalculateTicks ( wxDC & dc, mpWindow & w )
{
const int extend = w.GetScrX(); // /2;
// printf("this %p master %p\n", this, m_masterScale);
if ( m_masterScale )
{
@ -897,6 +971,9 @@ void mpScaleY::recalculateTicks ( wxDC & dc, mpWindow & w )
double minV, maxV, minVvis, maxVvis;
GetDataRange ( minV, maxV );
getVisibleDataRange ( w, minVvis, maxVvis );
//printf("vdr %.10f %.10f\n", minVvis, maxVvis);
m_absVisibleMaxV = std::max(std::abs(minVvis), std::abs(maxVvis));
m_tickValues.clear();
m_labeledTicks.clear();
@ -1036,7 +1113,7 @@ void mpScaleXLog::recalculateTicks ( wxDC & dc, mpWindow & w )
//printf("visibleD %.1f %.1f %.1f\n", visibleDecades, maxVvis, minVvis);
double d;
m_ticks.clear();
m_tickValues.clear();
m_labeledTicks.clear();
for ( d= minDecade; d<=maxDecade; d *= 10.0)
@ -1049,7 +1126,7 @@ void mpScaleXLog::recalculateTicks ( wxDC & dc, mpWindow & w )
if(visibleDecades < 2)
m_labeledTicks.push_back(dd);
//printf("dd %.1f\n",dd);
m_ticks.push_back(dd);
m_tickValues.push_back(dd);
}
}
@ -1079,7 +1156,7 @@ void mpScaleXLog::computeLabelExtents ( wxDC & dc, mpWindow & w )
int mpScaleXLog::tickCount() const
{
return m_ticks.size();
return m_tickValues.size();
}
int mpScaleXLog::labelCount() const
@ -1096,7 +1173,7 @@ const wxString mpScaleXLog::getLabel( int n )
double mpScaleXLog::getTickPos( int n )
{
return m_ticks[n]; //n0 + (double) n * step;
return m_tickValues[n]; //n0 + (double) n * step;
}
double mpScaleXLog::getLabelPos( int n )
@ -1132,6 +1209,10 @@ mpScaleXLog::mpScaleXLog(wxString name, int flags, bool ticks, unsigned int type
void mpScaleXBase::Plot(wxDC & dc, mpWindow & w)
{
int tx, ty;
m_offset = -m_minV;
m_scale = 1.0 / ( m_maxV - m_minV );
recalculateTicks( dc, w );
if (m_visible) {
@ -1667,6 +1748,11 @@ void mpWindow::OnMouseMove(wxMouseEvent &event)
dc.SetBrush(*wxTRANSPARENT_BRUSH);
dc.DrawRectangle(m_mouseLClick.x, m_mouseLClick.y, event.GetX() - m_mouseLClick.x, event.GetY() - m_mouseLClick.y);
m_zooming = true;
m_zoomRect.x = m_mouseLClick.x;
m_zoomRect.y = m_mouseLClick.y;
m_zoomRect.width = event.GetX() - m_mouseLClick.x;
m_zoomRect.height = event.GetY() - m_mouseLClick.y;
} else {
wxPoint moveVector(event.GetX() - m_mouseLClick.x, event.GetY() - m_mouseLClick.y);
m_movingInfoLayer->Move(moveVector);
@ -2194,6 +2280,14 @@ void mpWindow::OnPaint( wxPaintEvent& WXUNUSED(event) )
(*li)->Plot(*trgDc, *this);
};
if(m_zooming)
{
wxPen pen(m_fgColour, 1, wxDOT);
trgDc->SetPen(pen);
trgDc->SetBrush(*wxTRANSPARENT_BRUSH);
trgDc->DrawRectangle(m_zoomRect);
}
// If doublebuffer, draw now to the window:
if (m_enableDoubleBuffer)
{
@ -2797,6 +2891,7 @@ IMPLEMENT_DYNAMIC_CLASS(mpFXYVector, mpFXY)
mpFXYVector::mpFXYVector(wxString name, int flags ) : mpFXY(name,flags)
{
m_index = 0;
//printf("FXYVector::FXYVector!\n");
m_minX = -1;
m_maxX = 1;
m_minY = -1;
@ -2806,12 +2901,12 @@ mpFXYVector::mpFXYVector(wxString name, int flags ) : mpFXY(name,flags)
double mpScaleX::TransformToPlot ( double x )
{
return (x - m_minV) / (m_maxV - m_minV);
return (x + m_offset) * m_scale;
}
double mpScaleX::TransformFromPlot ( double xplot )
{
return xplot * (m_maxV - m_minV) + m_minV;
return xplot / m_scale - m_offset;
}
@ -2891,6 +2986,7 @@ void mpFXYVector::SetData( const std::vector<double> &xs,const std::vector<doubl
m_xs = xs;
m_ys = ys;
//printf("FXYVector::setData %d %d\n", xs.size(), ys.size());
// Update internal variables for the bounding box.
if (xs.size()>0)
@ -2912,6 +3008,9 @@ void mpFXYVector::SetData( const std::vector<double> &xs,const std::vector<doubl
if (*it<m_minY) m_minY=*it;
if (*it>m_maxY) m_maxY=*it;
}
//printf("minX %.10f maxX %.10f\n ", m_minX, m_maxX );
//printf("minY %.10f maxY %.10f\n ", m_minY, m_maxY );
}
else
{
@ -3483,3 +3582,20 @@ that correspond to the window point (dx0,dy0) within the image "m_bitmap", and
dc.DrawText( m_name, tx, ty);
}
}
void mpFXY::SetScale ( mpScaleBase *scaleX, mpScaleBase *scaleY )
{
m_scaleX = scaleX;
m_scaleY = scaleY;
//printf("SetScales : %p %p\n", scaleX, scaleY);
if(m_scaleX)
{
m_scaleX->ExtendDataRange(GetMinX(), GetMaxX());
}
if(m_scaleY)
{
m_scaleY->ExtendDataRange(GetMinY(), GetMaxY());
}
}

View File

@ -611,11 +611,7 @@ class WXDLLIMPEXP_MATHPLOT mpFXY : public mpLayer
*/
virtual void Plot(wxDC & dc, mpWindow & w);
virtual void SetScale ( mpScaleBase *scaleX, mpScaleBase *scaleY )
{
m_scaleX = scaleX;
m_scaleY = scaleY;
}
virtual void SetScale ( mpScaleBase *scaleX, mpScaleBase *scaleY );
protected:
@ -687,7 +683,7 @@ class WXDLLIMPEXP_MATHPLOT mpProfile : public mpLayer
class WXDLLIMPEXP_MATHPLOT mpScaleBase : public mpLayer
{
public:
mpScaleBase () {};
mpScaleBase () { m_rangeSet = false; };
virtual ~mpScaleBase () {};
bool HasBBox() { return FALSE; }
@ -710,6 +706,7 @@ public:
void SetDataRange ( double minV, double maxV )
{
m_rangeSet = true;
m_minV = minV;
m_maxV = maxV;
}
@ -720,11 +717,35 @@ public:
maxV = m_maxV;
}
void ExtendDataRange ( double minV, double maxV )
{
if(!m_rangeSet)
{
m_minV = minV;
m_maxV = maxV;
m_rangeSet = true;
} else {
m_minV = std::min(minV, m_minV);
m_maxV = std::max(maxV, m_maxV);
}
}
double AbsMaxValue() const
{
return std::max(std::abs(m_maxV), std::abs(m_minV));
}
double AbsVisibleMaxValue() const
{
return m_absVisibleMaxV;
}
virtual double TransformToPlot ( double x ) { return 0.0; };
virtual double TransformFromPlot (double xplot ){ return 0.0; };
protected:
virtual void getVisibleDataRange ( mpWindow& w, double &minV, double& maxV) {};
virtual void recalculateTicks ( wxDC & dc, mpWindow & w ) {};
virtual int tickCount() const { return 0; }
@ -733,9 +754,15 @@ protected:
virtual double getTickPos( int n ) { return 0.0; }
virtual double getLabelPos( int n ) { return 0.0; }
std::vector<double> m_tickValues;
std::vector<double> m_labeledTicks;
double m_offset, m_scale;
double m_absVisibleMaxV;
int m_flags; //!< Flag for axis alignment
bool m_ticks; //!< Flag to toggle between ticks or grid
double m_minV, m_maxV;
double m_rangeSet;
int m_maxLabelHeight;
int m_maxLabelWidth;
@ -828,8 +855,6 @@ class WXDLLIMPEXP_MATHPLOT mpScaleXLog : public mpScaleXBase
void computeLabelExtents ( wxDC & dc, mpWindow & w );
std::vector<double> m_ticks;
std::vector<double> m_labeledTicks;
double dig, step, end, n0, labelStep;
@ -901,8 +926,6 @@ class WXDLLIMPEXP_MATHPLOT mpScaleY : public mpScaleBase
std::vector<double> m_tickValues;
std::vector<double> m_labeledTicks;
double m_offset, m_scale;
int m_flags; //!< Flag for axis alignment
bool m_ticks; //!< Flag to toggle between ticks or grid
//wxString m_labelFormat; //!< Format string used to print labels
@ -1382,6 +1405,7 @@ class WXDLLIMPEXP_MATHPLOT mpWindow : public wxWindow
wxPoint m_scroll;
mpInfoLayer* m_movingInfoLayer; //!< For moving info layers over the window area
bool m_zooming;
wxRect m_zoomRect;
DECLARE_DYNAMIC_CLASS(mpWindow)
DECLARE_EVENT_TABLE()
};