QGIS/external/qwt-6.3.0/qwt_plot_rescaler.cpp
Juergen E. Fischer 33fc476d89 * replace external qwtpolar with qwt 6.3
* require qwt >=6.2 (and fallback to internal 6.3 if system's qwt doesn't suffice)
* debian doesn't have qwt for Qt6 and won't have it for trixie
2025-07-23 07:11:51 +10:00

655 lines
16 KiB
C++

/******************************************************************************
* Qwt Widget Library
* Copyright (C) 1997 Josef Wilgen
* Copyright (C) 2002 Uwe Rathmann
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the Qwt License, Version 1.0
*****************************************************************************/
#include "qwt_plot_rescaler.h"
#include "qwt_plot.h"
#include "qwt_scale_div.h"
#include "qwt_interval.h"
#include "qwt_plot_canvas.h"
#include <qevent.h>
class QwtPlotRescaler::AxisData
{
public:
AxisData()
: aspectRatio( 1.0 )
, expandingDirection( QwtPlotRescaler::ExpandUp )
{
}
double aspectRatio;
QwtInterval intervalHint;
QwtPlotRescaler::ExpandingDirection expandingDirection;
mutable QwtScaleDiv scaleDiv;
};
class QwtPlotRescaler::PrivateData
{
public:
PrivateData()
: referenceAxis( QwtAxis::XBottom )
, rescalePolicy( QwtPlotRescaler::Expanding )
, isEnabled( false )
, inReplot( 0 )
{
}
QwtPlotRescaler::AxisData* axisData( QwtAxisId axisId )
{
if ( !QwtAxis::isValid( axisId ) )
return NULL;
return &m_axisData[ axisId];
}
QwtAxisId referenceAxis;
RescalePolicy rescalePolicy;
bool isEnabled;
mutable int inReplot;
private:
QwtPlotRescaler::AxisData m_axisData[QwtAxis::AxisPositions];
};
/*!
Constructor
\param canvas Canvas
\param referenceAxis Reference axis, see RescalePolicy
\param policy Rescale policy
\sa setRescalePolicy(), setReferenceAxis()
*/
QwtPlotRescaler::QwtPlotRescaler( QWidget* canvas,
QwtAxisId referenceAxis, RescalePolicy policy )
: QObject( canvas )
{
m_data = new PrivateData;
m_data->referenceAxis = referenceAxis;
m_data->rescalePolicy = policy;
setEnabled( true );
}
//! Destructor
QwtPlotRescaler::~QwtPlotRescaler()
{
delete m_data;
}
/*!
\brief En/disable the rescaler
When enabled is true an event filter is installed for
the canvas, otherwise the event filter is removed.
\param on true or false
\sa isEnabled(), eventFilter()
*/
void QwtPlotRescaler::setEnabled( bool on )
{
if ( m_data->isEnabled != on )
{
m_data->isEnabled = on;
QWidget* w = canvas();
if ( w )
{
if ( m_data->isEnabled )
w->installEventFilter( this );
else
w->removeEventFilter( this );
}
}
}
/*!
\return true when enabled, false otherwise
\sa setEnabled, eventFilter()
*/
bool QwtPlotRescaler::isEnabled() const
{
return m_data->isEnabled;
}
/*!
Change the rescale policy
\param policy Rescale policy
\sa rescalePolicy()
*/
void QwtPlotRescaler::setRescalePolicy( RescalePolicy policy )
{
m_data->rescalePolicy = policy;
}
/*!
\return Rescale policy
\sa setRescalePolicy()
*/
QwtPlotRescaler::RescalePolicy QwtPlotRescaler::rescalePolicy() const
{
return m_data->rescalePolicy;
}
/*!
Set the reference axis ( see RescalePolicy )
\param axisId Axis
\sa referenceAxis()
*/
void QwtPlotRescaler::setReferenceAxis( QwtAxisId axisId )
{
m_data->referenceAxis = axisId;
}
/*!
\return Reference axis ( see RescalePolicy )
\sa setReferenceAxis()
*/
QwtAxisId QwtPlotRescaler::referenceAxis() const
{
return m_data->referenceAxis;
}
/*!
Set the direction in which all axis should be expanded
\param direction Direction
\sa expandingDirection()
*/
void QwtPlotRescaler::setExpandingDirection(
ExpandingDirection direction )
{
for ( int axis = 0; axis < QwtAxis::AxisPositions; axis++ )
setExpandingDirection( axis, direction );
}
/*!
Set the direction in which an axis should be expanded
\param axisId Axis
\param direction Direction
\sa expandingDirection()
*/
void QwtPlotRescaler::setExpandingDirection(
QwtAxisId axisId, ExpandingDirection direction )
{
if ( AxisData* axisData = m_data->axisData( axisId ) )
axisData->expandingDirection = direction;
}
/*!
\return Direction in which an axis should be expanded
\param axisId Axis
\sa setExpandingDirection()
*/
QwtPlotRescaler::ExpandingDirection
QwtPlotRescaler::expandingDirection( QwtAxisId axisId ) const
{
if ( const AxisData* axisData = m_data->axisData( axisId ) )
return axisData->expandingDirection;
return ExpandBoth;
}
/*!
Set the aspect ratio between the scale of the reference axis
and the other scales. The default ratio is 1.0
\param ratio Aspect ratio
\sa aspectRatio()
*/
void QwtPlotRescaler::setAspectRatio( double ratio )
{
for ( int axis = 0; axis < QwtAxis::AxisPositions; axis++ )
setAspectRatio( axis, ratio );
}
/*!
Set the aspect ratio between the scale of the reference axis
and another scale. The default ratio is 1.0
\param axisId Axis
\param ratio Aspect ratio
\sa aspectRatio()
*/
void QwtPlotRescaler::setAspectRatio( QwtAxisId axisId, double ratio )
{
if ( AxisData* axisData = m_data->axisData( axisId ) )
{
if ( ratio < 0.0 )
ratio = 0.0;
axisData->aspectRatio = ratio;
}
}
/*!
\return Aspect ratio between an axis and the reference axis.
\param axisId Axis
\sa setAspectRatio()
*/
double QwtPlotRescaler::aspectRatio( QwtAxisId axisId ) const
{
if ( AxisData* axisData = m_data->axisData( axisId ) )
return axisData->aspectRatio;
return 0.0;
}
/*!
Set an interval hint for an axis
In Fitting mode, the hint is used as minimal interval
that always needs to be displayed.
\param axisId Axis
\param interval Axis
\sa intervalHint(), RescalePolicy
*/
void QwtPlotRescaler::setIntervalHint( QwtAxisId axisId,
const QwtInterval& interval )
{
if ( AxisData* axisData = m_data->axisData( axisId ) )
axisData->intervalHint = interval;
}
/*!
\param axisId Axis
\return Interval hint
\sa setIntervalHint(), RescalePolicy
*/
QwtInterval QwtPlotRescaler::intervalHint( QwtAxisId axisId ) const
{
if ( AxisData* axisData = m_data->axisData( axisId ) )
return axisData->intervalHint;
return QwtInterval();
}
//! \return plot canvas
QWidget* QwtPlotRescaler::canvas()
{
return qobject_cast< QWidget* >( parent() );
}
//! \return plot canvas
const QWidget* QwtPlotRescaler::canvas() const
{
return qobject_cast< const QWidget* >( parent() );
}
//! \return plot widget
QwtPlot* QwtPlotRescaler::plot()
{
QWidget* w = canvas();
if ( w )
w = w->parentWidget();
return qobject_cast< QwtPlot* >( w );
}
//! \return plot widget
const QwtPlot* QwtPlotRescaler::plot() const
{
const QWidget* w = canvas();
if ( w )
w = w->parentWidget();
return qobject_cast< const QwtPlot* >( w );
}
//! Event filter for the plot canvas
bool QwtPlotRescaler::eventFilter( QObject* object, QEvent* event )
{
if ( object && object == canvas() )
{
switch ( event->type() )
{
case QEvent::Resize:
{
canvasResizeEvent( static_cast< QResizeEvent* >( event ) );
break;
}
case QEvent::PolishRequest:
{
rescale();
break;
}
default:;
}
}
return false;
}
/*!
Event handler for resize events of the plot canvas
\param event Resize event
\sa rescale()
*/
void QwtPlotRescaler::canvasResizeEvent( QResizeEvent* event )
{
const QMargins m = canvas()->contentsMargins();
const QSize marginSize( m.left() + m.right(), m.top() + m.bottom() );
const QSize newSize = event->size() - marginSize;
const QSize oldSize = event->oldSize() - marginSize;
rescale( oldSize, newSize );
}
//! Adjust the plot axes scales
void QwtPlotRescaler::rescale() const
{
const QSize size = canvas()->contentsRect().size();
rescale( size, size );
}
/*!
Adjust the plot axes scales
\param oldSize Previous size of the canvas
\param newSize New size of the canvas
*/
void QwtPlotRescaler::rescale(
const QSize& oldSize, const QSize& newSize ) const
{
if ( newSize.isEmpty() )
return;
QwtInterval intervals[QwtAxis::AxisPositions];
for ( int axisPos = 0; axisPos < QwtAxis::AxisPositions; axisPos++ )
{
const QwtAxisId axisId( axisPos );
intervals[axisPos] = interval( axisId );
}
const QwtAxisId refAxis = referenceAxis();
intervals[refAxis] = expandScale( refAxis, oldSize, newSize );
for ( int axisPos = 0; axisPos < QwtAxis::AxisPositions; axisPos++ )
{
const QwtAxisId axisId( axisPos );
if ( aspectRatio( axisId ) > 0.0 && axisId != refAxis )
{
intervals[axisPos] = syncScale(
axisId, intervals[refAxis], newSize );
}
}
updateScales( intervals );
}
/*!
Calculate the new scale interval of a plot axis
\param axisId Axis
\param oldSize Previous size of the canvas
\param newSize New size of the canvas
\return Calculated new interval for the axis
*/
QwtInterval QwtPlotRescaler::expandScale( QwtAxisId axisId,
const QSize& oldSize, const QSize& newSize ) const
{
const QwtInterval oldInterval = interval( axisId );
QwtInterval expanded = oldInterval;
switch ( rescalePolicy() )
{
case Fixed:
{
break; // do nothing
}
case Expanding:
{
if ( !oldSize.isEmpty() )
{
double width = oldInterval.width();
if ( orientation( axisId ) == Qt::Horizontal )
width *= double( newSize.width() ) / oldSize.width();
else
width *= double( newSize.height() ) / oldSize.height();
expanded = expandInterval( oldInterval,
width, expandingDirection( axisId ) );
}
break;
}
case Fitting:
{
double dist = 0.0;
for ( int axisPos = 0; axisPos < QwtAxis::AxisPositions; axisPos++ )
{
const QwtAxisId axisId( axisPos );
const double d = pixelDist( axisId, newSize );
if ( d > dist )
dist = d;
}
if ( dist > 0.0 )
{
double width;
if ( orientation( axisId ) == Qt::Horizontal )
width = newSize.width() * dist;
else
width = newSize.height() * dist;
expanded = expandInterval( intervalHint( axisId ),
width, expandingDirection( axisId ) );
}
break;
}
}
return expanded;
}
/*!
Synchronize an axis scale according to the scale of the reference axis
\param axisId Axis
\param reference Interval of the reference axis
\param size Size of the canvas
\return New interval for axis
*/
QwtInterval QwtPlotRescaler::syncScale( QwtAxisId axisId,
const QwtInterval& reference, const QSize& size ) const
{
double dist;
if ( orientation( referenceAxis() ) == Qt::Horizontal )
dist = reference.width() / size.width();
else
dist = reference.width() / size.height();
if ( orientation( axisId ) == Qt::Horizontal )
dist *= size.width();
else
dist *= size.height();
dist /= aspectRatio( axisId );
QwtInterval intv;
if ( rescalePolicy() == Fitting )
intv = intervalHint( axisId );
else
intv = interval( axisId );
intv = expandInterval( intv, dist, expandingDirection( axisId ) );
return intv;
}
/*!
\return Orientation of an axis
\param axisId Axis
*/
Qt::Orientation QwtPlotRescaler::orientation( QwtAxisId axisId ) const
{
return QwtAxis::isYAxis( axisId ) ? Qt::Vertical : Qt::Horizontal;
}
/*!
\param axisId Axis
\return Normalized interval of an axis
*/
QwtInterval QwtPlotRescaler::interval( QwtAxisId axisId ) const
{
if ( !plot()->isAxisValid( axisId ) )
return QwtInterval();
return plot()->axisScaleDiv( axisId ).interval().normalized();
}
/*!
Expand the interval
\param interval Interval to be expanded
\param width Distance to be added to the interval
\param direction Direction of the expand operation
\return Expanded interval
*/
QwtInterval QwtPlotRescaler::expandInterval(
const QwtInterval& interval, double width,
ExpandingDirection direction ) const
{
QwtInterval expanded = interval;
switch ( direction )
{
case ExpandUp:
expanded.setMinValue( interval.minValue() );
expanded.setMaxValue( interval.minValue() + width );
break;
case ExpandDown:
expanded.setMaxValue( interval.maxValue() );
expanded.setMinValue( interval.maxValue() - width );
break;
case ExpandBoth:
default:
expanded.setMinValue( interval.minValue() +
interval.width() / 2.0 - width / 2.0 );
expanded.setMaxValue( expanded.minValue() + width );
}
return expanded;
}
double QwtPlotRescaler::pixelDist( QwtAxisId axisId, const QSize& size ) const
{
const QwtInterval intv = intervalHint( axisId );
double dist = 0.0;
if ( !intv.isNull() )
{
if ( axisId == referenceAxis() )
{
dist = intv.width();
}
else
{
const double r = aspectRatio( axisId );
if ( r > 0.0 )
dist = intv.width() * r;
}
}
if ( dist > 0.0 )
{
if ( orientation( axisId ) == Qt::Horizontal )
dist /= size.width();
else
dist /= size.height();
}
return dist;
}
/*!
Update the axes scales
\param intervals Scale intervals
*/
void QwtPlotRescaler::updateScales(
QwtInterval intervals[QwtAxis::AxisPositions] ) const
{
if ( m_data->inReplot >= 5 )
{
return;
}
QwtPlot* plt = const_cast< QwtPlot* >( plot() );
const bool doReplot = plt->autoReplot();
plt->setAutoReplot( false );
for ( int axisPos = 0; axisPos < QwtAxis::AxisPositions; axisPos++ )
{
{
const QwtAxisId axisId( axisPos );
if ( axisId == referenceAxis() || aspectRatio( axisId ) > 0.0 )
{
double v1 = intervals[axisPos].minValue();
double v2 = intervals[axisPos].maxValue();
if ( !plt->axisScaleDiv( axisId ).isIncreasing() )
qSwap( v1, v2 );
if ( m_data->inReplot >= 1 )
m_data->axisData( axisId )->scaleDiv = plt->axisScaleDiv( axisId );
if ( m_data->inReplot >= 2 )
{
QList< double > ticks[QwtScaleDiv::NTickTypes];
for ( int t = 0; t < QwtScaleDiv::NTickTypes; t++ )
ticks[t] = m_data->axisData( axisId )->scaleDiv.ticks( t );
plt->setAxisScaleDiv( axisId, QwtScaleDiv( v1, v2, ticks ) );
}
else
{
plt->setAxisScale( axisId, v1, v2 );
}
}
}
}
QwtPlotCanvas* canvas = qobject_cast< QwtPlotCanvas* >( plt->canvas() );
bool immediatePaint = false;
if ( canvas )
{
immediatePaint = canvas->testPaintAttribute( QwtPlotCanvas::ImmediatePaint );
canvas->setPaintAttribute( QwtPlotCanvas::ImmediatePaint, false );
}
plt->setAutoReplot( doReplot );
m_data->inReplot++;
plt->replot();
m_data->inReplot--;
if ( canvas && immediatePaint )
{
canvas->setPaintAttribute( QwtPlotCanvas::ImmediatePaint, true );
}
}
#include "moc_qwt_plot_rescaler.cpp"