QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskScrollBox.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskScrollBox.h"
7#include "QskAnimationHint.h"
8#include "QskEvent.h"
9#include "QskFlickAnimator.h"
10#include "QskGesture.h"
11#include "QskPanGestureRecognizer.h"
12#include "QskQuick.h"
13#include "QskInternalMacros.h"
14
15QSK_QT_PRIVATE_BEGIN
16#include <private/qquickwindow_p.h>
17QSK_QT_PRIVATE_END
18
19static inline constexpr qreal qskViewportPadding()
20{
21 return 10.0; // should be from the skin, TODO ...
22}
23
24static inline bool qskIsScrollable(
25 const QskScrollBox* scrollBox, Qt::Orientations orientations )
26{
27 if ( orientations )
28 {
29 const QSizeF viewSize = scrollBox->viewContentsRect().size();
30 const QSizeF& scrollableSize = scrollBox->scrollableSize();
31
32 if ( orientations & Qt::Vertical )
33 {
34 if ( viewSize.height() < scrollableSize.height() )
35 return true;
36 }
37
38 if ( orientations & Qt::Horizontal )
39 {
40 if ( viewSize.width() < scrollableSize.width() )
41 return true;
42 }
43 }
44
45 return false;
46}
47
48namespace
49{
50 class FlickAnimator final : public QskFlickAnimator
51 {
52 public:
53 FlickAnimator()
54 {
55 // skin hints: TODO
56 setDuration( 1000 );
57 setEasingCurve( QEasingCurve::OutCubic );
58 }
59
60 void setScrollBox( QskScrollBox* scrollBox )
61 {
62 m_scrollBox = scrollBox;
63 }
64
65 void translate( qreal dx, qreal dy ) override
66 {
67 const QPointF pos = m_scrollBox->scrollPos();
68 m_scrollBox->setScrollPos( pos - QPointF( dx, -dy ) );
69 }
70
71 private:
72 QskScrollBox* m_scrollBox;
73 };
74
75 class ScrollAnimator final : public QskAnimator
76 {
77 public:
78 ScrollAnimator()
79 : m_scrollBox( nullptr )
80 {
81 }
82
83 void setScrollBox( QskScrollBox* scrollBox )
84 {
85 m_scrollBox = scrollBox;
86 }
87
88 void scroll( const QPointF& from, const QPointF& to )
89 {
90 if ( isRunning() )
91 {
92 m_to = to;
93 return;
94 }
95
96 if ( from == to || m_scrollBox == nullptr )
97 {
98 return;
99 }
100
101 m_from = from;
102 m_to = to;
103
104 const auto hint = m_scrollBox->flickHint();
105
106 setDuration( hint.duration );
107 setEasingCurve( hint.type );
108 setWindow( m_scrollBox->window() );
109
110 start();
111 }
112
113 protected:
114 void advance( qreal value ) override
115 {
116 qreal x = m_from.x() + ( m_to.x() - m_from.x() ) * value;
117 qreal y = m_from.y() + ( m_to.y() - m_from.y() ) * value;
118
119 m_scrollBox->setScrollPos( QPointF( x, y ) );
120 }
121
122 private:
123 QskScrollBox* m_scrollBox;
124
125 QPointF m_from;
126 QPointF m_to;
127 };
128
129 class PanRecognizer final : public QskPanGestureRecognizer
130 {
131 using Inherited = QskPanGestureRecognizer;
132
133 public:
134 PanRecognizer( QObject* parent = nullptr )
135 : QskPanGestureRecognizer( parent )
136 {
137 setOrientations( Qt::Horizontal | Qt::Vertical );
138 }
139
140 bool isAcceptedPos( const QPointF& pos ) const override
141 {
142 if ( auto scrollBox = qobject_cast< const QskScrollBox* >( watchedItem() ) )
143 {
144 if ( qskIsScrollable( scrollBox, orientations() ) )
145 return scrollBox->viewContentsRect().contains( pos );
146 }
147
148 return false;
149 }
150 };
151}
152
153class QskScrollBox::PrivateData
154{
155 public:
156 QPointF scrollPos;
157 QSizeF scrollableSize = QSize( 0.0, 0.0 );
158
159 PanRecognizer panRecognizer;
160
161 FlickAnimator flicker;
162 ScrollAnimator scroller;
163
164 bool autoScrollFocusItem = true;
165};
166
167QskScrollBox::QskScrollBox( QQuickItem* parent )
168 : Inherited( parent )
169 , m_data( new PrivateData() )
170{
171 m_data->flicker.setScrollBox( this );
172 m_data->scroller.setScrollBox( this );
173
174 m_data->panRecognizer.setWatchedItem( this );
175
176 setAcceptedMouseButtons( Qt::LeftButton );
177 setFiltersChildMouseEvents( true );
178
179 setWheelEnabled( true );
180 setFocusPolicy( Qt::StrongFocus );
181
182 connectWindow( window(), true );
183}
184
185QskScrollBox::~QskScrollBox()
186{
187}
188
189void QskScrollBox::setAutoScrollFocusedItem( bool on )
190{
191 if ( m_data->autoScrollFocusItem != on )
192 {
193 m_data->autoScrollFocusItem = on;
194 connectWindow( window(), true );
195 Q_EMIT autoScrollFocusedItemChanged( on );
196 }
197}
198
199bool QskScrollBox::autoScrollFocusItem() const
200{
201 return m_data->autoScrollFocusItem;
202}
203
204void QskScrollBox::onFocusItemChanged()
205{
206 if ( window() )
207 {
208#if QT_VERSION >= QT_VERSION_CHECK( 6, 1, 0 )
209 auto wd = QQuickWindowPrivate::get( window() )->deliveryAgentPrivate();
210#else
211 auto wd = QQuickWindowPrivate::get( window() );
212#endif
213 auto reason = wd->lastFocusReason;
214 if ( reason == Qt::TabFocusReason || reason == Qt::BacktabFocusReason )
215 ensureFocusItemVisible();
216 }
217}
218
219void QskScrollBox::ensureFocusItemVisible()
220{
221 if ( window() == nullptr )
222 return;
223
224 if ( const auto focusItem = window()->activeFocusItem() )
225 ensureItemVisible( focusItem );
226}
227
228void QskScrollBox::setFlickRecognizerTimeout( int timeout )
229{
230 if ( timeout < 0 )
231 timeout = -1;
232
233 m_data->panRecognizer.setTimeout( timeout );
234}
235
236int QskScrollBox::flickRecognizerTimeout() const
237{
238 return m_data->panRecognizer.timeout();
239}
240
241void QskScrollBox::setFlickableOrientations( Qt::Orientations orientations )
242{
243 if ( m_data->panRecognizer.orientations() != orientations )
244 {
245 m_data->panRecognizer.setOrientations( orientations );
246 Q_EMIT flickableOrientationsChanged();
247 }
248}
249
250Qt::Orientations QskScrollBox::flickableOrientations() const
251{
252 return m_data->panRecognizer.orientations();
253}
254
255void QskScrollBox::setScrollPos( const QPointF& pos )
256{
257 const QPointF boundedPos = boundedScrollPos( pos );
258 if ( boundedPos != m_data->scrollPos )
259 {
260 m_data->scrollPos = boundedPos;
261 update();
262
263 Q_EMIT scrollPosChanged();
264 Q_EMIT scrolledTo( boundedPos );
265 }
266}
267
268QPointF QskScrollBox::scrollPos() const
269{
270 return m_data->scrollPos;
271}
272
273void QskScrollBox::scrollTo( const QPointF& pos )
274{
275 m_data->scroller.scroll( scrollPos(), pos );
276}
277
278void QskScrollBox::setScrollableSize( const QSizeF& size )
279{
280 const QSizeF boundedSize = size.expandedTo( QSizeF( 0, 0 ) );
281
282 if ( boundedSize != m_data->scrollableSize )
283 {
284 m_data->scrollableSize = boundedSize;
285 Q_EMIT scrollableSizeChanged( m_data->scrollableSize );
286
287 setScrollPos( m_data->scrollPos ); // scroll pos might need to be re-bounded
288
289 update();
290 }
291}
292
293QSizeF QskScrollBox::scrollableSize() const
294{
295 return m_data->scrollableSize;
296}
297
298void QskScrollBox::ensureItemVisible( const QQuickItem* item )
299{
300 if ( qskIsAncestorOf( this, item ) )
301 {
302 auto pos = scrollPos() - viewContentsRect().topLeft();
303 pos += mapFromItem( item, QPointF() );
304
305 ensureVisible( QRectF( pos.x(), pos.y(), item->width(), item->height() ) );
306 }
307}
308
309void QskScrollBox::ensureVisible( const QPointF& pos )
310{
311 const qreal margin = qskViewportPadding();
312
313 QRectF r( scrollPos(), viewContentsRect().size() );
314 r.adjust( margin, margin, -margin, -margin );
315
316 qreal x = r.x();
317 qreal y = r.y();
318
319 if ( pos.x() < r.left() )
320 {
321 x = pos.x();
322 }
323 else if ( pos.x() > r.right() )
324 {
325 x = pos.x() - r.width();
326 }
327
328 if ( pos.y() < r.top() )
329 {
330 y = pos.y();
331 }
332 else if ( y > r.right() )
333 {
334 y = pos.y() - r.height();
335 }
336
337 const QPoint newPos( x - margin, y - margin );
338
339 if( isInitiallyPainted() && window() )
340 scrollTo( newPos );
341 else
342 setScrollPos( newPos );
343}
344
345void QskScrollBox::ensureVisible( const QRectF& itemRect )
346{
347 const qreal margin = qskViewportPadding();
348
349 QRectF r( scrollPos(), viewContentsRect().size() );
350 r.adjust( margin, margin, -margin, -margin );
351
352 qreal x = r.x();
353 qreal y = r.y();
354
355 if ( itemRect.width() > r.width() )
356 {
357 x = itemRect.left() + 0.5 * ( itemRect.width() - r.width() );
358 }
359 else if ( itemRect.right() > r.right() )
360 {
361 x = itemRect.right() - r.width();
362 }
363 else if ( itemRect.left() < r.left() )
364 {
365 x = itemRect.left();
366 }
367
368 if ( itemRect.height() > r.height() )
369 {
370 y = itemRect.top() + 0.5 * ( itemRect.height() - r.height() );
371 }
372 else if ( itemRect.bottom() > r.bottom() )
373 {
374 y = itemRect.bottom() - r.height();
375 }
376 else if ( itemRect.top() < r.top() )
377 {
378 y = itemRect.top();
379 }
380
381 const QPoint newPos( x - margin, y - margin );
382
383 if( isInitiallyPainted() && window() )
384 scrollTo( newPos );
385 else
386 setScrollPos( newPos );
387}
388
390{
392
393 connectWindow( event->oldWindow(), false );
394 connectWindow( event->window(), true );
395}
396
398{
399 if ( event->isResized() )
400 setScrollPos( scrollPos() );
401
403}
404
405void QskScrollBox::gestureEvent( QskGestureEvent* event )
406{
407 if ( event->gesture()->type() == QskGesture::Pan )
408 {
409 const auto gesture = static_cast< const QskPanGesture* >( event->gesture().get() );
410
411 switch ( gesture->state() )
412 {
413 case QskGesture::Updated:
414 {
415 setScrollPos( scrollPos() - gesture->delta() );
416 break;
417 }
418 case QskGesture::Finished:
419 {
420 m_data->flicker.setWindow( window() );
421 m_data->flicker.accelerate( gesture->angle(), gesture->velocity() );
422 break;
423 }
424 case QskGesture::Canceled:
425 {
426 // what to do here: maybe going back to the origin of the gesture ??
427 break;
428 }
429 default:
430 break;
431 }
432
433 return;
434 }
435
436 Inherited::gestureEvent( event );
437}
438
439#ifndef QT_NO_WHEELEVENT
440
441QPointF QskScrollBox::scrollOffset( const QWheelEvent* event ) const
442{
443 QPointF offset;
444
445 const auto pos = qskWheelPosition( event );
446 const auto viewRect = viewContentsRect();
447
448 if ( viewRect.contains( pos ) )
449 {
450 offset = event->pixelDelta();
451 if ( offset.isNull() )
452 offset = event->angleDelta() / QWheelEvent::DefaultDeltasPerStep;
453
454 offset.rx() *= viewRect.width();
455 offset.ry() *= viewRect.height();
456 }
457
458 return offset;
459}
460
461void QskScrollBox::wheelEvent( QWheelEvent* event )
462{
463 const auto offset = scrollOffset( event );
464 if ( !offset.isNull() )
465 setScrollPos( m_data->scrollPos - offset );
466}
467
468#endif
469
470QPointF QskScrollBox::boundedScrollPos( const QPointF& pos ) const
471{
472 const QRectF vr = viewContentsRect();
473
474 const qreal maxX = qMax( 0.0, scrollableSize().width() - vr.width() );
475 const qreal maxY = qMax( 0.0, scrollableSize().height() - vr.height() );
476
477 return QPointF( qBound( 0.0, pos.x(), maxX ), qBound( 0.0, pos.y(), maxY ) );
478}
479
480void QskScrollBox::connectWindow( const QQuickWindow* window, bool on )
481{
482 if ( ( window == nullptr ) || ( on && !autoScrollFocusItem() ) )
483 return;
484
485 if ( on )
486 {
487 QObject::connect( window, &QQuickWindow::activeFocusItemChanged,
488 this, &QskScrollBox::onFocusItemChanged, Qt::UniqueConnection );
489 }
490 else
491 {
492 QObject::disconnect( window, &QQuickWindow::activeFocusItemChanged,
493 this, &QskScrollBox::onFocusItemChanged );
494 }
495}
496
497#include "moc_QskScrollBox.cpp"
virtual void geometryChangeEvent(QskGeometryChangeEvent *)
Definition QskItem.cpp:871
bool isInitiallyPainted() const
Definition QskItem.cpp:584
virtual void windowChangeEvent(QskWindowChangeEvent *)
Definition QskItem.cpp:862
void windowChangeEvent(QskWindowChangeEvent *) override
void geometryChangeEvent(QskGeometryChangeEvent *) override