QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskFunctions.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskFunctions.h"
7#include "QskInternalMacros.h"
8
9#include <qfont.h>
10#include <qfontmetrics.h>
11#include <qmath.h>
12
13template< class Rect, class Value >
14static inline Rect qskAlignedRect( const Rect& outerRect,
15 Value width, Value height, Qt::Alignment alignment )
16{
17 Value x = outerRect.x();
18 Value y = outerRect.y();
19
20 switch ( alignment & Qt::AlignHorizontal_Mask )
21 {
22 case Qt::AlignHCenter:
23 x += ( outerRect.width() - width ) / 2;
24 break;
25
26 case Qt::AlignRight:
27 x += outerRect.width() - width;
28 break;
29
30 default:
31 break;
32 }
33
34 switch ( alignment & Qt::AlignVertical_Mask )
35 {
36 case Qt::AlignVCenter:
37 y += ( outerRect.height() - height ) / 2;
38 break;
39
40 case Qt::AlignBottom:
41 y += outerRect.height() - height;
42 break;
43
44 default:
45 break;
46 }
47
48 return Rect( x, y, width, height );
49}
50
51QRect qskAlignedRect( const QRect& outerRect,
52 int width, int height, Qt::Alignment alignment )
53{
54 return qskAlignedRect< QRect, int >( outerRect, width, height, alignment );
55}
56
57QRectF qskAlignedRectF( const QRectF& outerRect,
58 qreal width, qreal height, Qt::Alignment alignment )
59{
60 return qskAlignedRect< QRectF, qreal >( outerRect, width, height, alignment );
61}
62
63QRect qskInnerRect( const QRectF& rect )
64{
65 const int left = qCeil( rect.left() );
66 const int top = qCeil( rect.top() );
67 const int right = qFloor( rect.right() );
68 const int bottom = qFloor( rect.bottom() );
69
70 return QRect( left, top, right - left, bottom - top );
71}
72
73QRectF qskInnerRectF( const QRectF& rect )
74{
75 const qreal left = qCeil( rect.left() );
76 const qreal top = qCeil( rect.top() );
77 const qreal right = qFloor( rect.right() );
78 const qreal bottom = qFloor( rect.bottom() );
79
80 return QRectF( left, top, right - left, bottom - top );
81}
82
83QRectF qskValidOrEmptyInnerRect( const QRectF& rect, const QMarginsF& margins )
84{
85 qreal x, y, h, w;
86
87 if ( rect.width() > 0.0 )
88 {
89 const qreal marginsWidth = margins.left() + margins.right();
90
91 if ( marginsWidth > rect.width() )
92 {
93 x = rect.x() + rect.width() * ( margins.left() / marginsWidth );
94 w = 0.0;
95 }
96 else
97 {
98 x = rect.x() + margins.left();
99 w = rect.width() - marginsWidth;
100 }
101 }
102 else
103 {
104 x = rect.x();
105 w = 0.0;
106 }
107
108 if ( rect.height() > 0.0 )
109 {
110 const qreal marginsHeight = margins.top() + margins.bottom();
111 if ( marginsHeight > rect.height() )
112 {
113 y = rect.y() + rect.height() * ( margins.top() / marginsHeight );
114 h = 0.0;
115 }
116 else
117 {
118 y = rect.y() + margins.top();
119 h = rect.height() - marginsHeight;
120 }
121 }
122 else
123 {
124 y = rect.y();
125 h = 0.0;
126 }
127
128 return QRectF( x, y, w, h );
129}
130
131QRectF qskInterpolatedRect( const QRectF& from, const QRectF& to, qreal progress )
132{
133 if ( progress <= 0.0 )
134 return from;
135
136 if ( progress >= 1.0 )
137 return to;
138
139 const auto x = from.x() + progress * ( to.x() - from.x() );
140 const auto y = from.y() + progress * ( to.y() - from.y() );
141 const auto w = from.width() + progress * ( to.width() - from.width() );
142 const auto h = from.height() + progress * ( to.height() - from.height() );
143
144 return QRectF( x, y, w, h );
145}
146
147QSizeF qskInterpolatedSize( const QSizeF& from, const QSizeF& to, qreal progress )
148{
149 if ( progress <= 0.0 )
150 return from;
151
152 if ( progress >= 1.0 )
153 return to;
154
155 const auto w = from.width() + progress * ( to.width() - from.width() );
156 const auto h = from.height() + progress * ( to.height() - from.height() );
157
158 return QSizeF( w, h );
159}
160
161qreal qskHorizontalAdvance( const QFont& font, const QString& text )
162{
163 return qskHorizontalAdvance( QFontMetricsF( font ), text );
164}
165
166qreal qskHorizontalAdvance( const QFontMetricsF& fontMetrics, const QString& text )
167{
168 return fontMetrics.horizontalAdvance( text );
169}
170
171QSizeF qskTextRenderSize( const QFontMetricsF& fontMetrics, const QString& text )
172{
173 if ( text.isEmpty() )
174 return QSizeF( 0.0, 0.0 );
175
176 QRectF r( 0.0, 0.0, 10e6, 10e6 );
177 r = fontMetrics.boundingRect( r, 0, text );
178
179 return r.size();
180}
181
182qreal qskFuzzyFloor( qreal value, qreal stepSize )
183{
184 const double eps = 1.0e-6 * stepSize;
185
186 value = ( value + eps ) / stepSize;
187 return std::floor( value ) * stepSize;
188}
189
190qreal qskFuzzyCeil( qreal value, qreal stepSize )
191{
192 const double eps = 1.0e-6 * stepSize;
193
194 value = ( value - eps ) / stepSize;
195 return std::ceil( value ) * stepSize;
196}
197
198double qskConstrainedDegrees( double degrees )
199{
200 degrees = fmod( degrees, 360.0 );
201 if ( degrees < 0.0 )
202 degrees += 360.0;
203
204 return degrees;
205}
206
207float qskConstrainedDegrees( float degrees )
208{
209 degrees = fmodf( degrees, 360.0f );
210 if ( degrees < 0.0f )
211 degrees += 360.0f;
212
213 return degrees;
214}
215
216double qskConstrainedRadians( double radians )
217{
218 constexpr double pi2 = 2.0 * M_PI;
219
220 radians = fmod( radians, pi2 );
221 if ( radians < 0.0 )
222 radians += pi2;
223
224 return radians;
225}
226
227float qskConstrainedRadians( float radians )
228{
229 constexpr float pi2 = 2.0f * M_PI;
230
231 radians = fmodf( radians, pi2 );
232 if ( radians < 0.0f )
233 radians += pi2;
234
235 return radians;
236}
237
238QSK_HIDDEN_EXTERNAL_BEGIN
239
240bool qskHasEnvironment( const char* env )
241{
242 bool ok;
243
244 const int value = qEnvironmentVariableIntValue( env, &ok );
245 if ( ok )
246 return value != 0;
247
248 // All other strings are true, apart from "false"
249 auto result = qgetenv( env );
250 return !result.isEmpty() && result != "false";
251}
252
253QSK_HIDDEN_EXTERNAL_END