QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskGraduation.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskGraduation.h"
7#include "QskFunctions.h"
8#include "QskIntervalF.h"
9#include "QskTickmarks.h"
10
11#include <QDebug>
12#include <QtMath>
13
14#include <cmath>
15
16namespace
17{
18 /*
19 Code has been copied from the Qwt project ( with permission of the
20 copyright holder ( = me ). Maybe the code could be adjusted using Qskinny
21 fuzzy operators
22 */
23
24 class Engine
25 {
26 public:
27 static QskTickmarks buildTicks( const QskIntervalF&, qreal, int );
28 static QVector< qreal > buildMajorTicks( const QskIntervalF&, qreal );
29 static void buildMinorTicks( const QVector< qreal >&, int, qreal,
30 QVector< qreal >&, QVector< qreal >& );
31
32 private:
33 static bool fuzzyContains( const QskIntervalF&, double );
34 static int fuzzyCompare( double, double, double );
35 static double minorStepSize( double, int );
36
37 static QVector< qreal > strip( const QVector< qreal >&, const QskIntervalF& );
38 };
39
40 inline int Engine::fuzzyCompare( double value1, double value2, double intervalSize )
41 {
42 const double eps = std::abs( 1.0e-6 * intervalSize );
43
44 if ( value2 - value1 > eps )
45 return -1;
46
47 if ( value1 - value2 > eps )
48 return 1;
49
50 return 0;
51 }
52
53 inline bool Engine::fuzzyContains( const QskIntervalF& interval, double value )
54 {
55 if ( !interval.isValid() )
56 return false;
57
58 if ( fuzzyCompare( value, interval.lowerBound(), interval.length() ) < 0 )
59 return false;
60
61 if ( fuzzyCompare( value, interval.upperBound(), interval.length() ) > 0 )
62 return false;
63
64 return true;
65 }
66
67 double Engine::minorStepSize( double intervalSize, int maxSteps )
68 {
69 const double minStep = QskGraduation::stepSize( intervalSize, maxSteps );
70
71 if ( minStep != 0.0 )
72 {
73 // # ticks per interval
74 const int numTicks = qCeil( qAbs( intervalSize / minStep ) ) - 1;
75
76 // Do the minor steps fit into the interval?
77 if ( fuzzyCompare( ( numTicks + 1 ) * qAbs( minStep ),
78 qAbs( intervalSize ), intervalSize ) > 0 )
79 {
80 // The minor steps doesn't fit into the interval
81 return 0.5 * intervalSize;
82 }
83 }
84
85 return minStep;
86 }
87
88 QVector< qreal > Engine::strip(
89 const QVector< qreal >& ticks, const QskIntervalF& interval )
90 {
91 if ( !interval.isValid() || ticks.count() == 0 )
92 return QVector< qreal >();
93
94 if ( fuzzyContains( interval, ticks.first() )
95 && fuzzyContains( interval, ticks.last() ) )
96 {
97 return ticks;
98 }
99
100 QVector< qreal > strippedTicks;
101 strippedTicks.reserve( ticks.count() );
102
103 for ( const auto tick : ticks )
104 {
105 if ( fuzzyContains( interval, tick ) )
106 strippedTicks += tick;
107 }
108
109 return strippedTicks;
110 }
111
112 QVector< qreal > Engine::buildMajorTicks(
113 const QskIntervalF& interval, qreal stepSize )
114 {
115 int numTicks = qRound( interval.length() / stepSize ) + 1;
116 if ( numTicks > 10000 )
117 numTicks = 10000;
118
119 QVector< qreal > ticks;
120 ticks.reserve( numTicks );
121
122 ticks += interval.lowerBound();
123 for ( int i = 1; i < numTicks - 1; i++ )
124 ticks += interval.lowerBound() + i * stepSize;
125 ticks += interval.upperBound();
126
127 return ticks;
128 }
129
130 void Engine::buildMinorTicks(
131 const QVector< qreal >& majorTicks, int maxMinorSteps, qreal stepSize,
132 QVector< qreal >& minorTicks, QVector< qreal >& mediumTicks )
133 {
134 auto minStep = minorStepSize( stepSize, maxMinorSteps );
135 if ( minStep == 0.0 )
136 return;
137
138 // # ticks per interval
139 const int numTicks = qCeil( qAbs( stepSize / minStep ) ) - 1;
140
141 int medIndex = -1;
142 if ( numTicks % 2 )
143 medIndex = numTicks / 2;
144
145 // calculate minor ticks
146
147 for ( int i = 0; i < majorTicks.count(); i++ )
148 {
149 auto val = majorTicks[i];
150 for ( int k = 0; k < numTicks; k++ )
151 {
152 val += minStep;
153
154 double alignedValue = val;
155 if ( fuzzyCompare( val, 0.0, stepSize ) == 0 )
156 alignedValue = 0.0;
157
158 if ( k == medIndex )
159 mediumTicks += alignedValue;
160 else
161 minorTicks += alignedValue;
162 }
163 }
164 }
165
166 QskTickmarks Engine::buildTicks( const QskIntervalF& interval,
167 qreal stepSize, int maxMinorSteps )
168 {
169 using T = QskTickmarks;
170
171 const auto boundingInterval = interval.fuzzyAligned( stepSize );
172
173 QVector< qreal > ticks[3];
174 ticks[T::MajorTick] = buildMajorTicks( boundingInterval, stepSize );
175
176 if ( maxMinorSteps > 0 )
177 {
178 buildMinorTicks( ticks[T::MajorTick], maxMinorSteps, stepSize,
179 ticks[T::MinorTick], ticks[T::MediumTick] );
180 }
181
182 for ( auto& t : ticks )
183 {
184 t = strip( t, interval );
185
186 // ticks very close to 0.0 are
187 // explicitely set to 0.0
188
189 for ( int i = 0; i < t.count(); i++ )
190 {
191 if ( fuzzyCompare( t[i], 0.0, stepSize ) == 0 )
192 t[i] = 0.0;
193 }
194 }
195
196 return { ticks[T::MinorTick], ticks[T::MediumTick], ticks[T::MajorTick] };
197 }
198}
199
200QskTickmarks QskGraduation::divideInterval(
201 qreal x1, qreal x2, int maxMajorSteps, int maxMinorSteps, qreal stepSize )
202{
203 QskTickmarks tickmarks;
204
205 const auto interval = QskIntervalF::normalized( x1, x2 );
206
207 if ( interval.length() > std::numeric_limits< qreal >::max() )
208 {
209 qWarning() << "QskGraduation::divideInterval: overflow";
210 return tickmarks;
211 }
212
213 if ( interval.length() <= 0.0 || stepSize < 0.0 )
214 return tickmarks;
215
216 if ( stepSize == 0.0 )
217 {
218 if ( maxMajorSteps < 1 )
219 maxMajorSteps = 1;
220
221 stepSize = QskGraduation::stepSize( interval.length(), maxMajorSteps );
222 }
223
224 if ( stepSize != 0.0 )
225 tickmarks = Engine::buildTicks( interval, stepSize, maxMinorSteps );
226
227 return tickmarks;
228}
229
230qreal QskGraduation::stepSize( double length, int numSteps )
231{
232 if ( numSteps <= 0 )
233 return 0.0;
234
235 const auto v = length / numSteps;
236 if ( qFuzzyIsNull( v ) )
237 return 0.0;
238
239 constexpr double base = 10.0;
240
241 // the same as std::log10( std::fabs( v ) );
242 const double lx = std::log( std::fabs( v ) ) / std::log( base );
243 const double p = std::floor( lx );
244
245 const double fraction = std::pow( base, lx - p );
246
247 double stepSize = std::pow( base, p );
248 if ( v < 0 )
249 stepSize = -stepSize;
250
251 for ( const double f : { 2.0, 2.5, 5.0, 10.0 } )
252 {
253 if ( fraction <= f || qFuzzyCompare( fraction, f ) )
254 {
255 stepSize *= f;
256 break;
257 }
258 }
259
260 return stepSize;
261}