6#include "QskScrollBox.h"
7#include "QskAnimationHint.h"
9#include "QskFlickAnimator.h"
10#include "QskGesture.h"
11#include "QskPanGestureRecognizer.h"
13#include "QskInternalMacros.h"
16#include <private/qquickwindow_p.h>
19static inline constexpr qreal qskViewportPadding()
24static inline bool qskIsScrollable(
25 const QskScrollBox* scrollBox, Qt::Orientations orientations )
29 const QSizeF viewSize = scrollBox->viewContentsRect().size();
30 const QSizeF& scrollableSize = scrollBox->scrollableSize();
32 if ( orientations & Qt::Vertical )
34 if ( viewSize.height() < scrollableSize.height() )
38 if ( orientations & Qt::Horizontal )
40 if ( viewSize.width() < scrollableSize.width() )
57 setEasingCurve( QEasingCurve::OutCubic );
62 m_scrollBox = scrollBox;
65 void translate( qreal dx, qreal dy )
override
67 const QPointF pos = m_scrollBox->scrollPos();
68 m_scrollBox->setScrollPos( pos - QPointF( dx, -dy ) );
79 : m_scrollBox( nullptr )
85 m_scrollBox = scrollBox;
88 void scroll(
const QPointF& from,
const QPointF& to )
96 if ( from == to || m_scrollBox ==
nullptr )
104 const auto hint = m_scrollBox->flickHint();
106 setDuration( hint.duration );
107 setEasingCurve( hint.type );
108 setWindow( m_scrollBox->window() );
114 void advance( qreal value )
override
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;
119 m_scrollBox->setScrollPos( QPointF( x, y ) );
134 PanRecognizer( QObject* parent =
nullptr )
137 setOrientations( Qt::Horizontal | Qt::Vertical );
140 bool isAcceptedPos(
const QPointF& pos )
const override
142 if (
auto scrollBox = qobject_cast< const QskScrollBox* >( watchedItem() ) )
144 if ( qskIsScrollable( scrollBox, orientations() ) )
145 return scrollBox->viewContentsRect().contains( pos );
153class QskScrollBox::PrivateData
157 QSizeF scrollableSize = QSize( 0.0, 0.0 );
159 PanRecognizer panRecognizer;
161 FlickAnimator flicker;
162 ScrollAnimator scroller;
164 bool autoScrollFocusItem =
true;
167QskScrollBox::QskScrollBox( QQuickItem* parent )
168 : Inherited( parent )
169 , m_data( new PrivateData() )
171 m_data->flicker.setScrollBox(
this );
172 m_data->scroller.setScrollBox(
this );
174 m_data->panRecognizer.setWatchedItem(
this );
176 setAcceptedMouseButtons( Qt::LeftButton );
177 setFiltersChildMouseEvents(
true );
179 setWheelEnabled(
true );
180 setFocusPolicy( Qt::StrongFocus );
182 connectWindow( window(),
true );
185QskScrollBox::~QskScrollBox()
189void QskScrollBox::setAutoScrollFocusedItem(
bool on )
191 if ( m_data->autoScrollFocusItem != on )
193 m_data->autoScrollFocusItem = on;
194 connectWindow( window(),
true );
195 Q_EMIT autoScrollFocusedItemChanged( on );
199bool QskScrollBox::autoScrollFocusItem()
const
201 return m_data->autoScrollFocusItem;
204void QskScrollBox::onFocusItemChanged()
208#if QT_VERSION >= QT_VERSION_CHECK( 6, 1, 0 )
209 auto wd = QQuickWindowPrivate::get( window() )->deliveryAgentPrivate();
211 auto wd = QQuickWindowPrivate::get( window() );
213 auto reason = wd->lastFocusReason;
214 if ( reason == Qt::TabFocusReason || reason == Qt::BacktabFocusReason )
215 ensureFocusItemVisible();
219void QskScrollBox::ensureFocusItemVisible()
221 if ( window() ==
nullptr )
224 if (
const auto focusItem = window()->activeFocusItem() )
225 ensureItemVisible( focusItem );
228void QskScrollBox::setFlickRecognizerTimeout(
int timeout )
233 m_data->panRecognizer.setTimeout( timeout );
236int QskScrollBox::flickRecognizerTimeout()
const
238 return m_data->panRecognizer.timeout();
241void QskScrollBox::setFlickableOrientations( Qt::Orientations orientations )
243 if ( m_data->panRecognizer.orientations() != orientations )
245 m_data->panRecognizer.setOrientations( orientations );
246 Q_EMIT flickableOrientationsChanged();
250Qt::Orientations QskScrollBox::flickableOrientations()
const
252 return m_data->panRecognizer.orientations();
255void QskScrollBox::setScrollPos(
const QPointF& pos )
257 const QPointF boundedPos = boundedScrollPos( pos );
258 if ( boundedPos != m_data->scrollPos )
260 m_data->scrollPos = boundedPos;
263 Q_EMIT scrollPosChanged();
264 Q_EMIT scrolledTo( boundedPos );
268QPointF QskScrollBox::scrollPos()
const
270 return m_data->scrollPos;
273void QskScrollBox::scrollTo(
const QPointF& pos )
275 m_data->scroller.scroll( scrollPos(), pos );
278void QskScrollBox::setScrollableSize(
const QSizeF& size )
280 const QSizeF boundedSize = size.expandedTo( QSizeF( 0, 0 ) );
282 if ( boundedSize != m_data->scrollableSize )
284 m_data->scrollableSize = boundedSize;
285 Q_EMIT scrollableSizeChanged( m_data->scrollableSize );
287 setScrollPos( m_data->scrollPos );
293QSizeF QskScrollBox::scrollableSize()
const
295 return m_data->scrollableSize;
298void QskScrollBox::ensureItemVisible(
const QQuickItem* item )
300 if ( qskIsAncestorOf(
this, item ) )
302 auto pos = scrollPos() - viewContentsRect().topLeft();
303 pos += mapFromItem( item, QPointF() );
305 ensureVisible( QRectF( pos.x(), pos.y(), item->width(), item->height() ) );
309void QskScrollBox::ensureVisible(
const QPointF& pos )
311 const qreal margin = qskViewportPadding();
313 QRectF r( scrollPos(), viewContentsRect().size() );
314 r.adjust( margin, margin, -margin, -margin );
319 if ( pos.x() < r.left() )
323 else if ( pos.x() > r.right() )
325 x = pos.x() - r.width();
328 if ( pos.y() < r.top() )
332 else if ( y > r.right() )
334 y = pos.y() - r.height();
337 const QPoint newPos( x - margin, y - margin );
342 setScrollPos( newPos );
345void QskScrollBox::ensureVisible(
const QRectF& itemRect )
347 const qreal margin = qskViewportPadding();
349 QRectF r( scrollPos(), viewContentsRect().size() );
350 r.adjust( margin, margin, -margin, -margin );
355 if ( itemRect.width() > r.width() )
357 x = itemRect.left() + 0.5 * ( itemRect.width() - r.width() );
359 else if ( itemRect.right() > r.right() )
361 x = itemRect.right() - r.width();
363 else if ( itemRect.left() < r.left() )
368 if ( itemRect.height() > r.height() )
370 y = itemRect.top() + 0.5 * ( itemRect.height() - r.height() );
372 else if ( itemRect.bottom() > r.bottom() )
374 y = itemRect.bottom() - r.height();
376 else if ( itemRect.top() < r.top() )
381 const QPoint newPos( x - margin, y - margin );
386 setScrollPos( newPos );
393 connectWindow( event->oldWindow(),
false );
394 connectWindow( event->window(),
true );
399 if ( event->isResized() )
400 setScrollPos( scrollPos() );
407 if ( event->gesture()->type() == QskGesture::Pan )
409 const auto gesture =
static_cast< const QskPanGesture*
>(
event->gesture().get() );
411 switch ( gesture->state() )
413 case QskGesture::Updated:
415 setScrollPos( scrollPos() - gesture->delta() );
418 case QskGesture::Finished:
420 m_data->flicker.setWindow( window() );
421 m_data->flicker.accelerate( gesture->angle(), gesture->velocity() );
424 case QskGesture::Canceled:
436 Inherited::gestureEvent( event );
439#ifndef QT_NO_WHEELEVENT
441QPointF QskScrollBox::scrollOffset(
const QWheelEvent* event )
const
445 const auto pos = qskWheelPosition( event );
446 const auto viewRect = viewContentsRect();
448 if ( viewRect.contains( pos ) )
450 offset =
event->pixelDelta();
451 if ( offset.isNull() )
452 offset =
event->angleDelta() / QWheelEvent::DefaultDeltasPerStep;
454 offset.rx() *= viewRect.width();
455 offset.ry() *= viewRect.height();
461void QskScrollBox::wheelEvent( QWheelEvent* event )
463 const auto offset = scrollOffset( event );
464 if ( !offset.isNull() )
465 setScrollPos( m_data->scrollPos - offset );
470QPointF QskScrollBox::boundedScrollPos(
const QPointF& pos )
const
472 const QRectF vr = viewContentsRect();
474 const qreal maxX = qMax( 0.0, scrollableSize().width() - vr.width() );
475 const qreal maxY = qMax( 0.0, scrollableSize().height() - vr.height() );
477 return QPointF( qBound( 0.0, pos.x(), maxX ), qBound( 0.0, pos.y(), maxY ) );
480void QskScrollBox::connectWindow(
const QQuickWindow* window,
bool on )
482 if ( ( window ==
nullptr ) || ( on && !autoScrollFocusItem() ) )
487 QObject::connect( window, &QQuickWindow::activeFocusItemChanged,
488 this, &QskScrollBox::onFocusItemChanged, Qt::UniqueConnection );
492 QObject::disconnect( window, &QQuickWindow::activeFocusItemChanged,
493 this, &QskScrollBox::onFocusItemChanged );
497#include "moc_QskScrollBox.cpp"
virtual void geometryChangeEvent(QskGeometryChangeEvent *)
bool isInitiallyPainted() const
virtual void windowChangeEvent(QskWindowChangeEvent *)