6#include "QskStackBoxAnimator.h"
7#include "QskStackBox.h"
10#include "QskFunctions.h"
11#include "QskInternalMacros.h"
14#include <private/qquickitem_p.h>
19 class RotationTransform :
public QQuickTransform
24 RotationTransform( Qt::Axis axis, qreal radians, QQuickItem* item )
25 : QQuickTransform( item )
27 , m_radians( radians )
29 prependToItem( item );
32 void setRadians( qreal radians )
34 if ( m_radians != radians )
41 void applyTo( QMatrix4x4* matrix)
const override
43 if (
const auto item = qobject_cast< QQuickItem* >( parent() ) )
45 const auto dx = 0.5 * item->width();
46 const auto dy = 0.5 * item->height();
49 transform.translate( dx, dy );
50 transform.rotateRadians( m_radians, m_axis );
51 transform.translate( -dx, -dy );
58 const Qt::Axis m_axis;
62 class QuickTransform final :
public QQuickTransform
67 QuickTransform( QQuickItem* item )
68 : QQuickTransform( item )
70 prependToItem( item );
73 void setTransform(
const QTransform& transform )
75 if ( transform != m_transform )
77 m_transform = transform;
82 void applyTo( QMatrix4x4* matrix )
const override
84 if (
const auto item = qobject_cast< QQuickItem* >( parent() ) )
85 *matrix *= m_transform;
89 QTransform m_transform;
92 template<
typename Transform >
93 Transform* qskFindTransform(
const QQuickItem* item )
95 const auto& transforms = QQuickItemPrivate::get( item )->transforms;
96 for (
const auto& t : transforms )
98 if (
auto transform = qobject_cast< Transform* >( t ) )
106QskStackBoxAnimator::QskStackBoxAnimator(
QskStackBox* parent )
113QskStackBoxAnimator::~QskStackBoxAnimator()
117void QskStackBoxAnimator::setStartIndex(
int index )
119 m_transientIndex = m_startIndex = index;
122void QskStackBoxAnimator::setEndIndex(
int index )
127int QskStackBoxAnimator::startIndex()
const
132int QskStackBoxAnimator::endIndex()
const
142QQuickItem* QskStackBoxAnimator::itemAt(
int index )
const
144 return stackBox()->itemAtIndex(
145 ( index == 0 ) ? m_startIndex : m_endIndex );
148qreal QskStackBoxAnimator::transientIndex()
const
150 return m_transientIndex;
153void QskStackBoxAnimator::advance( qreal progress )
155 qreal transientIndex;
157 if ( qFuzzyIsNull( progress ) )
159 transientIndex = m_startIndex;
161 else if ( qFuzzyCompare( progress, 1.0 ) )
163 transientIndex = m_endIndex;
167 const auto box = stackBox();
169 auto startIndex = m_startIndex;
170 auto endIndex = m_endIndex;
172 if ( startIndex == 0 )
174 if ( endIndex == box->itemCount() - 1 )
175 startIndex += box->itemCount();
177 else if ( startIndex == box->itemCount() - 1 )
180 endIndex += box->itemCount();
183 transientIndex = startIndex + ( endIndex - startIndex ) * progress;
186 if ( !qskFuzzyCompare( m_transientIndex, transientIndex ) )
188 m_transientIndex = transientIndex;
189 advanceIndex( progress );
191 Q_EMIT stackBox()->transientIndexChanged( m_transientIndex );
195QskStackBoxAnimator1::QskStackBoxAnimator1(
QskStackBox* parent )
204QskStackBoxAnimator1::~QskStackBoxAnimator1()
208void QskStackBoxAnimator1::setDirection(
Qsk::Direction direction )
210 if ( m_direction != direction )
213 m_direction = direction;
222void QskStackBoxAnimator1::setup()
224 auto stackBox = this->stackBox();
226 m_hasClip = stackBox->clip();
228 stackBox->setClip(
true );
230 stackBox->installEventFilter(
this );
234void QskStackBoxAnimator1::advanceIndex( qreal value )
236 auto stackBox = this->stackBox();
241 for (
int i = 0; i < 2; i++ )
243 if (
auto item = itemAt( i ) )
245 QRectF rect = qskItemGeometry( item );
249 const int index = ( i == 0 ) ? startIndex() : endIndex();
250 rect = stackBox->geometryForItemAt( index );
252 m_itemOffset[ i ] = isHorizontal ? rect.x() : rect.y();
259 qreal off = stackBox->width() * ( value - i );
263 x = m_itemOffset[ i ] + off;
268 qreal off = stackBox->height() * ( value - i );
273 y = m_itemOffset[ i ] + off;
276 qskSetItemGeometry( item, x, y, rect.width(), rect.height() );
278 if ( !item->isVisible() )
279 item->setVisible(
true );
286void QskStackBoxAnimator1::done()
288 for (
int i = 0; i < 2; i++ )
290 if (
auto item = itemAt( i ) )
292 item->removeEventFilter(
this );
293 item->setVisible( i == 1 );
298 stackBox()->setClip(
false );
301bool QskStackBoxAnimator1::eventFilter( QObject*
object, QEvent* event )
303 if ( !m_isDirty &&
object == stackBox() )
305 switch(
static_cast< int >( event->type() ) )
307 case QskEvent::GeometryChange:
308 case QskEvent::ContentsRectChange:
309 case QskEvent::LayoutRequest:
317 return QObject::eventFilter(
object, event );
320QskStackBoxAnimator2::QskStackBoxAnimator2(
QskStackBox* parent )
322 , m_orientation( Qt::Horizontal )
323 , m_inverted( false )
327QskStackBoxAnimator2::~QskStackBoxAnimator2()
331void QskStackBoxAnimator2::setOrientation( Qt::Orientation orientation )
333 if ( m_orientation != orientation )
336 m_orientation = orientation;
340Qt::Orientation QskStackBoxAnimator2::orientation()
const
342 return static_cast< Qt::Orientation
>( m_orientation );
345void QskStackBoxAnimator2::setInverted(
bool on )
347 if ( m_inverted != on )
354bool QskStackBoxAnimator2::isInverted()
const
359void QskStackBoxAnimator2::setup()
361 const auto axis = ( m_orientation == Qt::Horizontal )
362 ? Qt::YAxis : Qt::XAxis;
364 if (
auto item = itemAt( 0 ) )
365 ( void )
new RotationTransform( axis, 0.0, item );
367 if (
auto item = itemAt( 1 ) )
368 (
void )
new RotationTransform( axis, M_PI_2, item );
371void QskStackBoxAnimator2::advanceIndex( qreal value )
373 auto radians = value * M_PI;
375 if ( radians < M_PI_2 && radians > -M_PI_2 )
377 if (
auto item = itemAt( 0 ) )
380 radians = 2 * M_PI - radians;
382 auto rotation = qskFindTransform< RotationTransform >( item );
383 rotation->setRadians( radians );
385 item->setVisible(
true );
388 if (
auto item = itemAt( 1 ) )
390 item->setVisible(
false );
395 if (
auto item = itemAt( 0 ) )
397 item->setVisible(
false );
400 if (
auto item = itemAt( 1 ) )
402 radians = radians - M_PI;
404 radians = 2 * M_PI - radians;
406 auto rotation = qskFindTransform< RotationTransform >( item );
407 rotation->setRadians( radians );
409 item->setVisible(
true );
414void QskStackBoxAnimator2::done()
416 for (
int i = 0; i < 2; i++ )
418 if (
auto item = itemAt( i ) )
420 delete qskFindTransform< RotationTransform >( item );
421 item->setVisible( i == 1 );
426QskStackBoxAnimator3::QskStackBoxAnimator3(
QskStackBox* parent )
431QskStackBoxAnimator3::~QskStackBoxAnimator3()
435void QskStackBoxAnimator3::setup()
437 if (
auto item = itemAt( 1 ) )
439 item->setOpacity( 0.0 );
440 item->setVisible(
true );
444void QskStackBoxAnimator3::advanceIndex( qreal value )
446 if (
auto item1 = itemAt( 0 ) )
447 item1->setOpacity( 1.0 - value );
449 if (
auto item2 = itemAt( 1 ) )
450 item2->setOpacity( value );
453void QskStackBoxAnimator3::done()
455 for (
int i = 0; i < 2; i++ )
457 if (
auto item = itemAt( i ) )
459 item->setOpacity( 1.0 );
460 item->setVisible( i == 1 );
465QskStackBoxAnimator4::QskStackBoxAnimator4(
QskStackBox* parent )
467 , m_orientation( Qt::Horizontal )
468 , m_inverted( false )
472QskStackBoxAnimator4::~QskStackBoxAnimator4()
476void QskStackBoxAnimator4::setOrientation( Qt::Orientation orientation )
478 if ( m_orientation != orientation )
481 m_orientation = orientation;
485Qt::Orientation QskStackBoxAnimator4::orientation()
const
487 return m_orientation;
490void QskStackBoxAnimator4::setInverted(
bool on )
492 if ( m_inverted != on )
499bool QskStackBoxAnimator4::isInverted()
const
504void QskStackBoxAnimator4::setup()
506 if (
auto item = itemAt( 0 ) )
508 ( void )
new QuickTransform( item );
509 item->setVisible(
true );
512 if (
auto item = itemAt( 1 ) )
514 ( void )
new QuickTransform( item );
515 item->setVisible(
true );
519void QskStackBoxAnimator4::advanceIndex( qreal value )
521 auto item1 = itemAt( 1 );
522 auto item2 = itemAt( 0 );
525 std::swap( item1, item2 );
527 const qreal posEdge = isInverted() ? value : 1.0 - value;
531 const auto transform = transformation( item1,
false, posEdge );
533 auto rotation = qskFindTransform< QuickTransform >( item1 );
534 rotation->setTransform( transform );
539 const auto transform = transformation( item2,
true, posEdge );
541 auto rotation = qskFindTransform< QuickTransform >( item2 );
542 rotation->setTransform( transform );
546QTransform QskStackBoxAnimator4::transformation(
547 const QQuickItem* item,
bool first, qreal posEdge )
const
555 const qreal radians = M_PI_2 * ( 1.0 - posEdge );
557 QTransform transform;
559 if( orientation() == Qt::Horizontal )
561 const qreal dx = posEdge * ( item->x() + item->width() );
562 const qreal dy = 0.5 * item->height();
566 transform.translate( -item->x() + dx, dy );
567 transform.rotateRadians( radians, Qt::YAxis );
568 transform.translate( -item->width(), -dy );
572 transform.translate( dx, dy );
573 transform.rotateRadians( radians - M_PI_2, Qt::YAxis );
574 transform.translate( 0.0, -dy );
579 const qreal dx = 0.5 * item->width();
580 const qreal dy = posEdge * ( item->y() + item->height() );
584 transform.translate( dx, -item->y() + dy );
585 transform.rotateRadians( radians, Qt::XAxis );
586 transform.translate( -dx, -item->height() );
590 transform.translate( dx, dy );
591 transform.rotateRadians( radians - M_PI_2, Qt::XAxis );
592 transform.translate( -dx, 0.0 );
599void QskStackBoxAnimator4::done()
601 for (
int i = 0; i < 2; i++ )
603 if (
auto item = itemAt( i ) )
605 delete qskFindTransform< QuickTransform >( item );
606 item->setVisible( i == 1 );
611#include "moc_QskStackBoxAnimator.cpp"
612#include "QskStackBoxAnimator.moc"
Direction
This enum type specifies a horizontal ot vertical direction.