QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskProgressBarSkinlet.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskProgressBarSkinlet.h"
7#include "QskProgressBar.h"
8#include "QskIntervalF.h"
9#include "QskBoxBorderMetrics.h"
10
11#include <qeasingcurve.h>
12#include <cmath>
13
14using Q = QskProgressBar;
15
16static QskIntervalF qskFillInterval( const QskProgressIndicator* indicator )
17{
18 qreal pos1, pos2;
19
20 if ( indicator->isIndeterminate() )
21 {
22 const auto pos = indicator->positionHint( Q::Fill );
23
24 static const QEasingCurve curve( QEasingCurve::InOutCubic );
25
26 const qreal off = 0.15;
27
28 pos1 = curve.valueForProgress( qMax( pos - off, 0.0 ) );
29 pos2 = curve.valueForProgress( qMin( pos + off, 1.0 ) );
30 }
31 else
32 {
33 pos1 = indicator->valueAsRatio( indicator->origin() );
34 pos2 = indicator->valueAsRatio( indicator->value() );
35 }
36
37 auto bar = static_cast< const QskProgressBar* >( indicator );
38 if( bar->orientation() == Qt::Horizontal )
39 {
40 if ( bar->layoutMirroring() )
41 {
42 pos1 = 1.0 - pos1;
43 pos2 = 1.0 - pos2;
44 }
45 }
46
47 if ( pos1 > pos2 )
48 std::swap( pos1, pos2 );
49
50 return QskIntervalF( pos1, pos2 );
51}
52
53static QskGradient qskFillGradient( const QskProgressBar* progressBar )
54{
55 auto gradient = progressBar->gradientHint( Q::Fill );
56
57 if ( gradient.isVisible() && !gradient.isMonochrome()
58 && ( gradient.type() == QskGradient::Stops ) )
59 {
60 /*
61 When having stops only we use a linear gradient,
62 where the colors are increasing in direction of the
63 progress value. We interprete the gradient as a
64 definition for the 100% situation and have to adjust
65 the stops for smaller bars.
66
67 For this situation it would be more convenient to
68 adjust the start/stop positions, but the box renderer is
69 not supporting this yet. TODO ...
70 */
71
72 const auto intv = qskFillInterval( progressBar );
73
74 const auto stops = qskExtractedGradientStops(
75 gradient.stops(), intv.lowerBound(), intv.upperBound() );
76
77 gradient.setStops( stops );
78
79 gradient.setLinearDirection( progressBar->orientation() );
80
81 if ( progressBar->orientation() == Qt::Vertical || progressBar->layoutMirroring() )
82 gradient.reverse();
83 }
84
85 return gradient;
86}
87
88QskProgressBarSkinlet::QskProgressBarSkinlet( QskSkin* skin )
89 : Inherited( skin )
90{
91}
92
93QskProgressBarSkinlet::~QskProgressBarSkinlet()
94{
95}
96
97QRectF QskProgressBarSkinlet::subControlRect(
98 const QskSkinnable* skinnable, const QRectF& contentsRect,
99 QskAspect::Subcontrol subControl ) const
100{
101 const auto bar = static_cast< const Q* >( skinnable );
102
103 if( subControl == Q::Groove )
104 return grooveRect( bar, contentsRect );
105
106 if( subControl == Q::Fill )
107 return fillRect( bar );
108
109 return Inherited::subControlRect( skinnable, contentsRect, subControl );
110}
111
112QSGNode* QskProgressBarSkinlet::updateGrooveNode(
113 const QskProgressIndicator* indicator, QSGNode* node ) const
114{
115 return updateBoxNode( indicator, node, Q::Groove );
116}
117
118QSGNode* QskProgressBarSkinlet::updateFillNode(
119 const QskProgressIndicator* indicator, QSGNode* node ) const
120{
121 const auto rect = indicator->subControlRect( Q::Fill );
122 if ( rect.isEmpty() )
123 return nullptr;
124
125 const auto progressBar = static_cast< const Q* >( indicator );
126 return updateBoxNode( indicator, node, rect,
127 qskFillGradient( progressBar ), Q::Fill );
128}
129
130QRectF QskProgressBarSkinlet::grooveRect(
131 const QskProgressBar* progressBar, const QRectF& contentsRect ) const
132{
133 const auto size = progressBar->metric( Q::Groove | QskAspect::Size );
134
135 auto rect = contentsRect;
136 if ( progressBar->orientation() == Qt::Horizontal )
137 {
138 rect.setY( rect.y() + 0.5 * ( rect.height() - size ) );
139 rect.setHeight( size );
140 }
141 else
142 {
143 rect.setX( rect.x() + 0.5 * ( rect.width() - size ) );
144 rect.setWidth( size );
145 }
146
147 return rect;
148}
149
150QRectF QskProgressBarSkinlet::fillRect( const QskProgressBar* progressBar ) const
151{
152 const auto size = progressBar->metric( Q::Fill | QskAspect::Size );
153 auto rect = progressBar->subControlRect( Q::Groove );
154
155 if ( progressBar->orientation() == Qt::Horizontal )
156 {
157 rect.setY( rect.y() + 0.5 * ( rect.height() - size ) );
158 rect.setHeight( size );
159 }
160 else
161 {
162 rect.setX( rect.x() + 0.5 * ( rect.width() - size ) );
163 rect.setWidth( size );
164 }
165
166 const auto borderMetrics = progressBar->boxBorderMetricsHint( Q::Groove );
167
168 auto m = progressBar->paddingHint( Q::Groove );
169 m += 0.5 * borderMetrics.toAbsolute( rect.size() ).widths();
170
171 rect = rect.marginsRemoved( m );
172
173 const auto intv = qskFillInterval( progressBar );
174
175 if( progressBar->orientation() == Qt::Horizontal )
176 {
177 const auto w = rect.width();
178
179 rect.setRight( rect.left() + intv.upperBound() * w );
180 rect.setLeft( rect.left() + intv.lowerBound() * w );
181 }
182 else
183 {
184 const auto h = rect.height();
185
186 rect.setTop( rect.bottom() - h * intv.upperBound() );
187 rect.setBottom( rect.bottom() - h * intv.lowerBound() );
188 }
189
190 return rect;
191}
192
193QSizeF QskProgressBarSkinlet::sizeHint( const QskSkinnable* skinnable,
194 Qt::SizeHint which, const QSizeF& ) const
195{
196 if ( which != Qt::PreferredSize )
197 return QSizeF();
198
199 auto extent = skinnable->metric( Q::Groove | QskAspect::Size );
200 extent = qMax( extent, skinnable->metric( Q::Fill | QskAspect::Size ) );
201
202 const auto progressBar = static_cast< const QskProgressBar* >( skinnable );
203
204 if ( progressBar->orientation() == Qt::Horizontal )
205 return QSizeF( -1, extent );
206 else
207 return QSizeF( extent, -1 );
208}
209
210#include "moc_QskProgressBarSkinlet.cpp"
Subcontrol
For use within the rendering or lay-outing of a specific QskSkinnable.
Definition QskAspect.h:104
QRectF subControlRect(QskAspect::Subcontrol) const
bool layoutMirroring() const
Definition QskItem.cpp:517
Base class for progress indicators.
QMarginsF paddingHint(QskAspect, QskSkinHintStatus *=nullptr) const
Retrieves a padding hint.
QskGradient gradientHint(QskAspect, QskSkinHintStatus *=nullptr) const
Retrieves a color hint as gradient.
QskBoxBorderMetrics boxBorderMetricsHint(QskAspect, QskSkinHintStatus *=nullptr) const
Retrieves a border hint.