6#include "QskGraduation.h"
7#include "QskFunctions.h"
8#include "QskIntervalF.h"
9#include "QskTickmarks.h"
28 static QVector< qreal > buildMajorTicks(
const QskIntervalF&, qreal );
29 static void buildMinorTicks(
const QVector< qreal >&,
int, qreal,
30 QVector< qreal >&, QVector< qreal >& );
33 static bool fuzzyContains(
const QskIntervalF&,
double );
34 static int fuzzyCompare(
double,
double,
double );
35 static double minorStepSize(
double,
int );
37 static QVector< qreal > strip(
const QVector< qreal >&,
const QskIntervalF& );
40 inline int Engine::fuzzyCompare(
double value1,
double value2,
double intervalSize )
42 const double eps = std::abs( 1.0e-6 * intervalSize );
44 if ( value2 - value1 > eps )
47 if ( value1 - value2 > eps )
53 inline bool Engine::fuzzyContains(
const QskIntervalF& interval,
double value )
55 if ( !interval.isValid() )
58 if ( fuzzyCompare( value, interval.lowerBound(), interval.length() ) < 0 )
61 if ( fuzzyCompare( value, interval.upperBound(), interval.length() ) > 0 )
67 double Engine::minorStepSize(
double intervalSize,
int maxSteps )
69 const double minStep = QskGraduation::stepSize( intervalSize, maxSteps );
74 const int numTicks = qCeil( qAbs( intervalSize / minStep ) ) - 1;
77 if ( fuzzyCompare( ( numTicks + 1 ) * qAbs( minStep ),
78 qAbs( intervalSize ), intervalSize ) > 0 )
81 return 0.5 * intervalSize;
88 QVector< qreal > Engine::strip(
89 const QVector< qreal >& ticks,
const QskIntervalF& interval )
91 if ( !interval.isValid() || ticks.count() == 0 )
92 return QVector< qreal >();
94 if ( fuzzyContains( interval, ticks.first() )
95 && fuzzyContains( interval, ticks.last() ) )
100 QVector< qreal > strippedTicks;
101 strippedTicks.reserve( ticks.count() );
103 for (
const auto tick : ticks )
105 if ( fuzzyContains( interval, tick ) )
106 strippedTicks += tick;
109 return strippedTicks;
112 QVector< qreal > Engine::buildMajorTicks(
115 int numTicks = qRound( interval.length() / stepSize ) + 1;
116 if ( numTicks > 10000 )
119 QVector< qreal > ticks;
120 ticks.reserve( numTicks );
122 ticks += interval.lowerBound();
123 for (
int i = 1; i < numTicks - 1; i++ )
124 ticks += interval.lowerBound() + i * stepSize;
125 ticks += interval.upperBound();
130 void Engine::buildMinorTicks(
131 const QVector< qreal >& majorTicks,
int maxMinorSteps, qreal stepSize,
132 QVector< qreal >& minorTicks, QVector< qreal >& mediumTicks )
134 auto minStep = minorStepSize( stepSize, maxMinorSteps );
135 if ( minStep == 0.0 )
139 const int numTicks = qCeil( qAbs( stepSize / minStep ) ) - 1;
143 medIndex = numTicks / 2;
147 for (
int i = 0; i < majorTicks.count(); i++ )
149 auto val = majorTicks[i];
150 for (
int k = 0; k < numTicks; k++ )
154 double alignedValue = val;
155 if ( fuzzyCompare( val, 0.0, stepSize ) == 0 )
159 mediumTicks += alignedValue;
161 minorTicks += alignedValue;
167 qreal stepSize,
int maxMinorSteps )
171 const auto boundingInterval = interval.fuzzyAligned( stepSize );
173 QVector< qreal > ticks[3];
174 ticks[T::MajorTick] = buildMajorTicks( boundingInterval, stepSize );
176 if ( maxMinorSteps > 0 )
178 buildMinorTicks( ticks[T::MajorTick], maxMinorSteps, stepSize,
179 ticks[T::MinorTick], ticks[T::MediumTick] );
182 for (
auto& t : ticks )
184 t = strip( t, interval );
189 for (
int i = 0; i < t.count(); i++ )
191 if ( fuzzyCompare( t[i], 0.0, stepSize ) == 0 )
196 return { ticks[T::MinorTick], ticks[T::MediumTick], ticks[T::MajorTick] };
201 qreal x1, qreal x2,
int maxMajorSteps,
int maxMinorSteps, qreal stepSize )
205 const auto interval = QskIntervalF::normalized( x1, x2 );
207 if ( interval.length() > std::numeric_limits< qreal >::max() )
209 qWarning() <<
"QskGraduation::divideInterval: overflow";
213 if ( interval.length() <= 0.0 || stepSize < 0.0 )
216 if ( stepSize == 0.0 )
218 if ( maxMajorSteps < 1 )
221 stepSize = QskGraduation::stepSize( interval.length(), maxMajorSteps );
224 if ( stepSize != 0.0 )
225 tickmarks = Engine::buildTicks( interval, stepSize, maxMinorSteps );
230qreal QskGraduation::stepSize(
double length,
int numSteps )
235 const auto v = length / numSteps;
236 if ( qFuzzyIsNull( v ) )
239 constexpr double base = 10.0;
242 const double lx = std::log( std::fabs( v ) ) / std::log( base );
243 const double p = std::floor( lx );
245 const double fraction = std::pow( base, lx - p );
247 double stepSize = std::pow( base, p );
249 stepSize = -stepSize;
251 for (
const double f : { 2.0, 2.5, 5.0, 10.0 } )
253 if ( fraction <= f || qFuzzyCompare( fraction, f ) )