QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskRgbValue.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskRgbValue.h"
7#include "QskInternalMacros.h"
8
9#include <qeasingcurve.h>
10#include <qimage.h>
11
12QSK_QT_PRIVATE_BEGIN
13#include <private/qdrawhelper_p.h>
14QSK_QT_PRIVATE_END
15
16namespace
17{
18 inline int value( int from, int to, qreal ratio )
19 {
20 return int( from + ( to - from ) * ratio );
21 }
22
23 inline qreal valueF( qreal from, qreal to, qreal ratio )
24 {
25 return int( from + ( to - from ) * ratio );
26 }
27}
28
29static inline QColor qskInterpolatedColor(
30 const QColor& c1, const QColor& c2, qreal ratio )
31{
32 switch ( c1.spec() )
33 {
34 case QColor::Rgb:
35 {
36 const int r = value( c1.red(), c2.red(), ratio );
37 const int g = value( c1.green(), c2.green(), ratio );
38 const int b = value( c1.blue(), c2.blue(), ratio );
39 const int a = value( c1.alpha(), c2.alpha(), ratio );
40
41 return QColor::fromRgb( r, g, b, a );
42 }
43 case QColor::Hsv:
44 {
45 const int h = value( c1.hue(), c2.hue(), ratio );
46 const int s = value( c1.saturation(), c2.saturation(), ratio );
47 const int v = value( c1.value(), c2.value(), ratio );
48 const int a = value( c1.alpha(), c2.alpha(), ratio );
49
50 return QColor::fromHsv( h, s, v, a );
51 }
52 case QColor::Cmyk:
53 {
54 const int c = value( c1.cyan(), c2.cyan(), ratio );
55 const int m = value( c1.magenta(), c2.magenta(), ratio );
56 const int y = value( c1.yellow(), c2.yellow(), ratio );
57 const int k = value( c1.black(), c2.black(), ratio );
58 const int a = value( c1.alpha(), c2.alpha(), ratio );
59
60 return QColor::fromCmykF( c, m, y, k, a );
61 }
62 case QColor::Hsl:
63 {
64 const int h = value( c1.hue(), c2.hue(), ratio );
65 const int s = value( c1.saturation(), c2.saturation(), ratio );
66 const int l = value( c1.lightness(), c2.lightness(), ratio );
67 const int a = value( c1.alpha(), c2.alpha(), ratio );
68
69 return QColor::fromHsl( h, s, l, a );
70 }
71 case QColor::ExtendedRgb:
72 {
73 const qreal r = valueF( c1.redF(), c2.redF(), ratio );
74 const qreal g = valueF( c1.greenF(), c2.greenF(), ratio );
75 const qreal b = valueF( c1.blueF(), c2.blueF(), ratio );
76 const qreal a = valueF( c1.alphaF(), c2.alphaF(), ratio );
77
78 return QColor::fromRgbF( r, g, b, a );
79 }
80 case QColor::Invalid:
81 break;
82 }
83
84 return c2;
85}
86
87QRgb QskRgb::interpolated( QRgb rgb1, QRgb rgb2, qreal ratio )
88{
89 // interpolating in HSV usually provides better results !!
90
91 if ( rgb1 == rgb2 )
92 return rgb1;
93
94 const int r = value( qRed( rgb1 ), qRed( rgb2 ), ratio );
95 const int g = value( qGreen( rgb1 ), qGreen( rgb2 ), ratio );
96 const int b = value( qBlue( rgb1 ), qBlue( rgb2 ), ratio );
97 const int a = value( qAlpha( rgb1 ), qAlpha( rgb2 ), ratio );
98
99 return qRgba( r, g, b, a );
100}
101
102QColor QskRgb::interpolated( const QColor& c1, const QColor& c2, qreal ratio )
103{
104 if ( c1 == c2 )
105 return c2;
106
107 /*
108 If one of the colors is invalid we treat it like
109 a transparent version of the other color
110 */
111
112 if ( !c1.isValid() )
113 {
114 QColor c = c2;
115 c.setAlpha( ratio * c2.alpha() );
116 return c;
117 }
118
119 if ( !c2.isValid() )
120 {
121 QColor c = c1;
122 c.setAlpha( ( 1.0 - ratio ) * c1.alpha() );
123 return c;
124 }
125
126 if ( c1.spec() == c2.spec() )
127 return qskInterpolatedColor( c1, c2, ratio );
128 else
129 return qskInterpolatedColor( c1.convertTo( c2.spec() ), c2, ratio );
130}
131
132QRgb QskRgb::rgb( Qt::GlobalColor color )
133{
134 using namespace QskRgb;
135
136 static constexpr QRgb rgbValues[] =
137 {
138 White, // Qt::color0
139 Black, // Qt::color1
140 Black, // Qt::black
141 White, // Qt::white
142 Grey, // Qt::darkGray
143 qRgb( 160, 160, 164 ), // Qt::gray
144 Silver, // Qt::lightGray
145 Red, // Qt::red
146 Lime, // Qt::green
147 Blue, // Qt::blue
148 Cyan, // Qt::cyan
149 Magenta, // Qt::magenta
150 Yellow, // Qt::yellow
151 Maroon, // Qt::darkRed
152 Green, // Qt::darkGreen
153 Navy, // Qt::darkBlue
154 Teal, // Qt::darkCyan
155 Purple, // Qt::darkMagenta
156 Olive, // Qt::darkYellow
157 Transparent // Qt::transparent
158 };
159
160 return rgbValues[ color ];
161}
162
163QRgb QskRgb::lighter( QRgb rgb, int factor ) noexcept
164{
165 if ( factor <= 0 )
166 return rgb;
167
168 // guess we can find a faster implementation without using QColor TODO ...
169 return QColor::fromRgba( rgb ).lighter( factor ).rgba();
170}
171
172QRgb QskRgb::darker( QRgb rgb, int factor ) noexcept
173{
174 if ( factor <= 0 )
175 return rgb;
176
177 // guess we can find a faster implementation without using QColor TODO ...
178 return QColor::fromRgba( rgb ).darker( factor ).rgba();
179}
180
181#ifndef QT_NO_DEBUG_STREAM
182
183#include <qdebug.h>
184
185void QskRgb::debugColor( QDebug debug, const QColor& color )
186{
187 debugColor( debug, color.rgba() );
188}
189
190void QskRgb::debugColor( QDebug debug, QRgb rgb )
191{
192 QDebugStateSaver saver( debug );
193 debug.nospace();
194
195 debug << '[';
196
197 debug << qRed( rgb ) << "r,"
198 << qGreen( rgb ) << "g," << qBlue( rgb ) << 'b';
199
200 if ( qAlpha( rgb ) != 255 )
201 debug << ',' << qAlpha( rgb ) << 'a';
202
203 debug << ']';
204}
205
206#endif
207
208QImage QskRgb::colorTable( int size, const QskGradientStops& stops )
209{
210 if ( size == 0 || stops.isEmpty() )
211 return QImage();
212
213 QImage image( size, 1, QImage::Format_RGBA8888_Premultiplied );
214
215 if ( stops.size() == 1 )
216 {
217 const auto rgb = ARGB2RGBA( qPremultiply( stops[0].rgb() ) );
218 image.fill( rgb );
219
220 return image;
221 }
222
223 auto values = reinterpret_cast< uint* >( image.bits() );
224
225 int index1, index2;
226 QRgb rgb1, rgb2;
227
228 index1 = index2 = qRound( stops[0].position() * size );
229 rgb1 = rgb2 = qPremultiply( stops[0].rgb() );
230
231 if ( index1 > 0 )
232 {
233 const auto v = ARGB2RGBA( rgb1 );
234
235 for ( int i = 0; i < index1; i++ )
236 values[i] = v;
237 }
238
239 for ( int i = 1; i < stops.count(); i++ )
240 {
241 const auto& stop = stops[i];
242
243 index2 = qRound( stop.position() * size );
244 rgb2 = qPremultiply( stop.rgb() );
245
246 const auto n = index2 - index1;
247
248 values[ index1 ] = ARGB2RGBA( rgb1 );
249 for ( int j = 1; j < n; j++ )
250 {
251 const auto rgb = QskRgb::interpolated( rgb1, rgb2, qreal( j ) / ( n - 1 ) );
252 values[ index1 + j] = ARGB2RGBA( rgb );
253 }
254
255 index1 = index2;
256 rgb1 = rgb2;
257 }
258
259 if ( index1 < size - 1 )
260 {
261 const auto v = ARGB2RGBA( rgb1 );
262
263 for ( int i = index1; i < size ; i++ )
264 values[i] = v;
265 }
266
267 return image;
268}
269
270QImage QskRgb::colorTable( const int size,
271 QRgb rgb1, QRgb rgb2, const QEasingCurve& curve )
272{
273 if ( size == 0 )
274 return QImage();
275
276 rgb1 = qPremultiply( rgb1 );
277 rgb2 = qPremultiply( rgb2 );
278
279 QImage image( size, 1, QImage::Format_RGBA8888_Premultiplied );
280
281 if ( rgb1 == rgb2 )
282 {
283 image.fill( ARGB2RGBA( rgb1 ) );
284 return image;
285 }
286
287 auto values = reinterpret_cast< uint* >( image.bits() );
288
289 for ( int i = 0; i < size; i++ )
290 {
291 qreal progress = curve.valueForProgress( qreal( i ) / ( size - 1 ) );
292 progress = qBound( 0.0, progress, 1.0 );
293
294 auto rgb = QskRgb::interpolated( rgb1, rgb2, progress );
295 values[i] = ARGB2RGBA( rgb );
296 }
297
298 return image;
299}