QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskProgressRingSkinlet.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskProgressRingSkinlet.h"
7#include "QskArcMetrics.h"
8#include "QskProgressRing.h"
9#include "QskIntervalF.h"
10
11using Q = QskProgressRing;
12
13static QskIntervalF qskFillInterval( const QskProgressIndicator* indicator )
14{
15 qreal pos1, pos2;
16
17 if ( indicator->isIndeterminate() )
18 {
19 pos1 = indicator->positionHint( QskProgressIndicator::Fill );
20 pos2 = 2 * pos1;
21 }
22 else
23 {
24 pos1 = indicator->valueAsRatio( indicator->origin() );
25 pos2 = indicator->valueAsRatio( indicator->value() );
26 }
27
28 if ( pos1 > pos2 )
29 std::swap( pos1, pos2 );
30
31 return QskIntervalF( pos1, pos2 );
32}
33
34static inline QPair< qreal, qreal > qskFillAngles(
35 const QskArcMetrics& metrics, const QskIntervalF& intv )
36{
37 const auto startAngle = metrics.startAngle() + intv.lowerBound() * metrics.spanAngle();
38 const auto endAngle = metrics.startAngle() + intv.upperBound() * metrics.spanAngle();
39
40 return { startAngle, endAngle };
41}
42
43QskProgressRingSkinlet::QskProgressRingSkinlet( QskSkin* skin )
44 : Inherited( skin )
45{
46}
47
48QskProgressRingSkinlet::~QskProgressRingSkinlet()
49{
50}
51
52QRectF QskProgressRingSkinlet::subControlRect(
53 const QskSkinnable* skinnable, const QRectF& contentsRect,
54 QskAspect::Subcontrol subControl ) const
55{
56 if( subControl == Q::Groove || subControl == Q::Fill )
57 return contentsRect;
58
59 return Inherited::subControlRect( skinnable, contentsRect, subControl );
60}
61
62QSGNode* QskProgressRingSkinlet::updateGrooveNode(
63 const QskProgressIndicator* indicator, QSGNode* node ) const
64{
65 const auto ring = static_cast< const Q* >( indicator );
66
67 const auto spacing = ring->spacingHint( Q::Fill ); // degrees
68
69 if( spacing > 0.0 )
70 {
71 const auto fillMetrics = ring->arcMetricsHint( Q::Fill );
72 if ( fillMetrics.isClosed() )
73 {
74 const auto fillAngles = qskFillAngles( fillMetrics, qskFillInterval( ring ) );
75
76 qreal startAngle, endAngle;
77 if ( fillAngles.second > fillAngles.first )
78 {
79 startAngle = fillAngles.second + spacing;
80 endAngle = fillAngles.first + 360.0 - spacing;
81 }
82 else
83 {
84 startAngle = fillAngles.second - spacing;
85 endAngle = fillAngles.first - 360.0 + spacing;
86 }
87
88 return updateArcNode( ring, node,
89 startAngle, endAngle - startAngle, Q::Groove );
90 }
91 }
92
93 return updateArcNode( indicator, node, Q::Groove );
94}
95
96QSGNode* QskProgressRingSkinlet::updateFillNode(
97 const QskProgressIndicator* indicator, QSGNode* node ) const
98{
99 const auto ring = static_cast< const Q* >( indicator );
100
101 const auto subControl = Q::Fill;
102
103 const auto rect = ring->subControlRect( subControl );
104 if ( rect.isEmpty() )
105 return nullptr;
106
107 const auto metrics = ring->arcMetricsHint( subControl );
108 if ( metrics.isNull() )
109 return nullptr;
110
111 auto gradient = ring->gradientHint( subControl );
112 if ( !gradient.isVisible() )
113 return nullptr;
114
115 const auto intv = qskFillInterval( ring );
116
117 if ( ( gradient.type() == QskGradient::Stops ) && !gradient.isMonochrome() )
118 {
119 const auto stops = qskExtractedGradientStops( gradient.stops(),
120 intv.lowerBound(), intv.upperBound() );
121
122 gradient.setStops( stops );
123
124 if ( metrics.spanAngle() < 0.0 )
125 gradient.reverse();
126 }
127
128 const auto angles = qskFillAngles( metrics, intv );
129
130 return updateArcNode( ring, node, rect, gradient,
131 angles.first, angles.second - angles.first, subControl );
132}
133
134QSizeF QskProgressRingSkinlet::sizeHint( const QskSkinnable* skinnable,
135 Qt::SizeHint which, const QSizeF& constraint ) const
136{
137 if ( which != Qt::PreferredSize )
138 return QSizeF();
139
140 auto hint = skinnable->strutSizeHint( Q::Fill );
141 hint = hint.expandedTo( skinnable->strutSizeHint( Q::Groove ) );
142
143 if ( !constraint.isEmpty() )
144 {
145 const qreal aspectRatio = hint.isEmpty() ? 1.0 : hint.width() / hint.height();
146
147 if ( constraint.width() >= 0.0 )
148 hint.setHeight( constraint.width() / aspectRatio );
149 else
150 hint.setWidth( constraint.height() * aspectRatio );
151 }
152
153 return hint;
154}
155
156#include "moc_QskProgressRingSkinlet.cpp"
Subcontrol
For use within the rendering or lay-outing of a specific QskSkinnable.
Definition QskAspect.h:104
QRectF subControlRect(QskAspect::Subcontrol) const
Base class for progress indicators.
static const QskAspect::Subcontrol Fill
Circular progress indicator.
qreal spacingHint(QskAspect, QskSkinHintStatus *=nullptr) const
Retrieves a spacing hint.
QSizeF strutSizeHint(QskAspect, QskSkinHintStatus *=nullptr) const
Retrieves a strut size hint.