QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskSlider.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskSlider.h"
7#include "QskAnimationHint.h"
8#include "QskAspect.h"
9#include "QskEvent.h"
10#include "QskFunctions.h"
11
12QSK_SUBCONTROL( QskSlider, Panel )
13QSK_SUBCONTROL( QskSlider, Groove )
14QSK_SUBCONTROL( QskSlider, Fill )
15QSK_SUBCONTROL( QskSlider, Scale )
16QSK_SUBCONTROL( QskSlider, Tick )
17QSK_SUBCONTROL( QskSlider, Handle )
18
19QSK_SYSTEM_STATE( QskSlider, Pressed, QskAspect::FirstSystemState << 2 )
20
21static QRectF qskHandleSelectionRect( const QskSlider* slider )
22{
23 return slider->subControlRect( QskSlider::Handle );
24}
25
26static QRectF qskSliderSelectionRect( const QskSlider* slider )
27{
28 const qreal margin = 10.0;
29
30 const auto scaleRect = slider->subControlRect( QskSlider::Scale );
31 const auto handleRect = qskHandleSelectionRect( slider );
32
33 auto r = slider->subControlRect( QskSlider::Panel );
34 if ( slider->orientation() == Qt::Horizontal )
35 {
36 r.setTop( qMin( r.top(), handleRect.top() ) );
37 r.setBottom( qMax( r.bottom(), handleRect.bottom() ) );
38 r.setLeft( scaleRect.left() - margin );
39 r.setRight( scaleRect.right() + margin );
40 }
41 else
42 {
43 r.setLeft( qMin( r.left(), handleRect.left() ) );
44 r.setRight( qMax( r.right(), handleRect.right() ) );
45 r.setTop( scaleRect.top() - margin );
46 r.setBottom( scaleRect.bottom() + margin );
47 }
48
49 return r;
50}
51
52static inline int qskKeyOffset( Qt::Orientation orientation, int key )
53{
54 if ( orientation == Qt::Horizontal )
55 {
56 if ( key == Qt::Key_Left )
57 return -1;
58
59 if ( key == Qt::Key_Right )
60 return 1;
61 }
62 else
63 {
64 if ( key == Qt::Key_Down )
65 return -1;
66
67 if ( key == Qt::Key_Up )
68 return 1;
69 }
70
71 return 0;
72}
73
74class QskSlider::PrivateData
75{
76 public:
77 PrivateData( Qt::Orientation orientation )
78 : pressedValue( 0 )
79 , hasOrigin( false )
80 , inverted( false )
81 , tracking( true )
82 , dragging( false )
83 , orientation( orientation )
84 {
85 }
86
87 QPointF pressedPos;
88 qreal pressedValue;
89
90 qreal origin = 0.0;
91
92 bool hasOrigin : 1;
93 bool inverted : 1;
94 bool tracking : 1;
95 bool dragging : 1;
96 uint orientation : 2;
97};
98
99QskSlider::QskSlider( QQuickItem* parent )
100 : QskSlider( Qt::Horizontal, parent )
101{
102}
103
104QskSlider::QskSlider( Qt::Orientation orientation, QQuickItem* parent )
105 : Inherited( parent )
106 , m_data( new PrivateData( orientation ) )
107{
108 setAcceptHoverEvents( true );
109 setFocusPolicy( Qt::StrongFocus );
110
111 if ( orientation == Qt::Horizontal )
112 initSizePolicy( QskSizePolicy::Minimum, QskSizePolicy::Fixed );
113 else
114 initSizePolicy( QskSizePolicy::Fixed, QskSizePolicy::Minimum );
115
116 connect( this, &QskSlider::boundariesChanged, this, &QskSlider::moveHandle );
117 connect( this, &QskSlider::valueChanged, this, &QskSlider::moveHandle );
118}
119
120QskSlider::~QskSlider()
121{
122}
123
124void QskSlider::setOrientation( Qt::Orientation orientation )
125{
126 if ( orientation != m_data->orientation )
127 {
128 m_data->orientation = orientation;
129#if 1
130 // swapping the size policy: guess this is what a user expects
131 setSizePolicy( sizePolicy( Qt::Vertical ), sizePolicy( Qt::Horizontal ) );
132#endif
134 update();
135
136 Q_EMIT orientationChanged( this->orientation() );
137 }
138}
139
140Qt::Orientation QskSlider::orientation() const
141{
142 return static_cast< Qt::Orientation >( m_data->orientation );
143}
144
145void QskSlider::setInverted( bool on )
146{
147 if ( on != m_data->inverted )
148 {
149 m_data->inverted = on;
150 update();
151
152 Q_EMIT invertedChanged( on );
153 }
154}
155
156bool QskSlider::isInverted() const
157{
158 return m_data->inverted;
159}
160
161void QskSlider::setOrigin( qreal origin )
162{
163 if ( isComponentComplete() )
164 origin = boundedValue( origin );
165
166 if( !m_data->hasOrigin || !qskFuzzyCompare( m_data->origin, origin ) )
167 {
168 m_data->hasOrigin = true;
169 m_data->origin = origin;
170
171 update();
172 Q_EMIT originChanged( origin );
173 }
174}
175
176void QskSlider::resetOrigin()
177{
178 if ( m_data->hasOrigin )
179 {
180 m_data->hasOrigin = false;
181
182 update();
183 Q_EMIT originChanged( origin() );
184 }
185}
186
187qreal QskSlider::origin() const
188{
189 if ( m_data->hasOrigin )
190 return boundedValue( m_data->origin );
191
192 return minimum();
193}
194
195bool QskSlider::hasOrigin() const
196{
197 return m_data->hasOrigin;
198}
199
201{
202 return static_cast< QskAspect::Variation >( m_data->orientation );
203}
204
205void QskSlider::setTracking( bool on )
206{
207 if ( on != m_data->tracking )
208 {
209 m_data->tracking = on;
210 Q_EMIT trackingChanged( on );
211 }
212}
213
214bool QskSlider::isTracking() const
215{
216 return m_data->tracking;
217}
218
219void QskSlider::componentComplete()
220{
221 Inherited::componentComplete();
222 if ( m_data->hasOrigin )
223 m_data->origin = boundedValue( m_data->origin );
224}
225
227{
228 setPositionHint( Handle, valueAsRatio() );
230}
231
232void QskSlider::mousePressEvent( QMouseEvent* event )
233{
234 const auto pos = qskMousePosition( event );
235 if ( !qskHandleSelectionRect( this ).contains( pos ) )
236 {
237 const auto r = qskSliderSelectionRect( this );
238 if ( !r.contains( pos ) )
239 {
240 Inherited::mousePressEvent( event );
241 return;
242 }
243
244 qreal ratio;
245
246 const auto scaleRect = subControlRect( Scale );
247 if ( m_data->orientation == Qt::Horizontal )
248 ratio = ( pos.x() - scaleRect.left() ) / scaleRect.width();
249 else
250 ratio = ( scaleRect.bottom() - pos.y() ) / scaleRect.height();
251
252 if ( m_data->inverted )
253 ratio = 1.0 - ratio;
254
255 setValue( valueFromRatio( ratio ) );
256 }
257
258 setSkinStateFlag( Pressed );
259
260 m_data->pressedPos = pos;
261 m_data->pressedValue = value();
262}
263
264void QskSlider::mouseMoveEvent( QMouseEvent* event )
265{
266 if ( !hasSkinState( Pressed ) )
267 return;
268
269 const auto mousePos = qskMousePosition( event );
270 const auto r = subControlRect( Scale );
271
272 auto length = boundaryLength();
273 if ( m_data->inverted )
274 length = -length;
275
276 qreal newValue;
277
278 if ( m_data->orientation == Qt::Horizontal )
279 {
280 const auto distance = mousePos.x() - m_data->pressedPos.x();
281 newValue = m_data->pressedValue + distance / r.width() * length;
282 }
283 else
284 {
285 const auto distance = mousePos.y() - m_data->pressedPos.y();
286 newValue = m_data->pressedValue - distance / r.height() * length;
287 }
288
289 if ( m_data->tracking )
290 {
291 m_data->dragging = true;
292 setValue( newValue );
293 m_data->dragging = false;
294 }
295 else
296 {
297 // moving the handle without changing the value
298 moveHandleTo( newValue, QskAnimationHint() );
299 }
300}
301
302void QskSlider::mouseReleaseEvent( QMouseEvent* )
303{
304 if ( !m_data->tracking && ( m_data->pressedValue != value() ) )
305 Q_EMIT valueChanged( value() );
306
307 setSkinStateFlag( Pressed, false );
308}
309
310void QskSlider::keyPressEvent( QKeyEvent* event )
311{
312 if ( auto offset = qskKeyOffset( orientation(), event->key() ) )
313 {
314 if ( m_data->inverted )
315 offset = -offset;
316
317 increment( offset * stepSize() );
318 return;
319 }
320
321 if ( m_data->hasOrigin )
322 {
323 switch( event->key() )
324 {
325 case Qt::Key_Home:
326 {
327 setValue( origin() );
328 return;
329 }
330
331 case Qt::Key_End:
332 {
333 // we have 2 endpoints - better do nothing
334 return;
335 }
336 }
337 }
338
339 Inherited::keyPressEvent( event );
340}
341
342void QskSlider::moveHandle()
343{
344 QskAnimationHint hint;
345 if ( !m_data->dragging )
346 {
347 const auto aspect = Handle | QskAspect::Metric | QskAspect::Position;
348 hint = animationHint( aspect | skinStates() );
349 }
350
351 moveHandleTo( value(), hint );
352}
353
354void QskSlider::moveHandleTo( qreal value, const QskAnimationHint& hint )
355{
356 const qreal pos = valueAsRatio( value );
357
358 if ( hint.isValid() )
359 {
360 const qreal oldPos = positionHint( Handle );
361 setPositionHint( Handle, pos );
362
363 const auto aspect = Handle | QskAspect::Metric | QskAspect::Position;
364 startTransition( aspect, hint, oldPos, pos );
365 }
366 else
367 {
368 setPositionHint( Handle, pos );
369 }
370
371 update();
372}
373
374#include "moc_QskSlider.cpp"
@ FirstSystemState
Definition QskAspect.h:115
Variation
Some sort of variation.
Definition QskAspect.h:82
QRectF subControlRect(QskAspect::Subcontrol) const
void setSizePolicy(QskSizePolicy)
QskSizePolicy sizePolicy
Definition QskControl.h:43
void resetImplicitSize()
Definition QskItem.cpp:727
virtual void aboutToShow()
Definition QskItem.cpp:1099
QskAnimationHint animationHint(QskAspect, QskSkinHintStatus *=nullptr) const
void setSkinStateFlag(QskAspect::State, bool on=true)
QskAspect::Variation effectiveVariation() const override
void aboutToShow() override