QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskStackBoxAnimator.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskStackBoxAnimator.h"
7#include "QskStackBox.h"
8#include "QskEvent.h"
9#include "QskQuick.h"
10#include "QskFunctions.h"
11#include "QskInternalMacros.h"
12
13QSK_QT_PRIVATE_BEGIN
14#include <private/qquickitem_p.h>
15QSK_QT_PRIVATE_END
16
17namespace
18{
19 class RotationTransform : public QQuickTransform
20 {
21 Q_OBJECT
22
23 public:
24 RotationTransform( Qt::Axis axis, qreal radians, QQuickItem* item )
25 : QQuickTransform( item )
26 , m_axis( axis )
27 , m_radians( radians )
28 {
29 prependToItem( item );
30 }
31
32 void setRadians( qreal radians )
33 {
34 if ( m_radians != radians )
35 {
36 m_radians = radians;
37 update();
38 }
39 }
40
41 void applyTo( QMatrix4x4* matrix) const override
42 {
43 if ( const auto item = qobject_cast< QQuickItem* >( parent() ) )
44 {
45 const auto dx = 0.5 * item->width();
46 const auto dy = 0.5 * item->height();
47
48 QTransform transform;
49 transform.translate( dx, dy );
50 transform.rotateRadians( m_radians, m_axis );
51 transform.translate( -dx, -dy );
52
53 *matrix *= transform;
54 }
55 }
56
57 private:
58 const Qt::Axis m_axis;
59 qreal m_radians;
60 };
61
62 class QuickTransform final : public QQuickTransform
63 {
64 Q_OBJECT
65
66 public:
67 QuickTransform( QQuickItem* item )
68 : QQuickTransform( item )
69 {
70 prependToItem( item );
71 }
72
73 void setTransform( const QTransform& transform )
74 {
75 if ( transform != m_transform )
76 {
77 m_transform = transform;
78 update();
79 }
80 }
81
82 void applyTo( QMatrix4x4* matrix ) const override
83 {
84 if ( const auto item = qobject_cast< QQuickItem* >( parent() ) )
85 *matrix *= m_transform;
86 }
87
88 private:
89 QTransform m_transform;
90 };
91
92 template< typename Transform >
93 Transform* qskFindTransform( const QQuickItem* item )
94 {
95 const auto& transforms = QQuickItemPrivate::get( item )->transforms;
96 for ( const auto& t : transforms )
97 {
98 if ( auto transform = qobject_cast< Transform* >( t ) )
99 return transform;
100 }
101
102 return nullptr;
103 }
104}
105
106QskStackBoxAnimator::QskStackBoxAnimator( QskStackBox* parent )
107 : QObject( parent )
108 , m_startIndex( -1 )
109 , m_endIndex( -1 )
110{
111}
112
113QskStackBoxAnimator::~QskStackBoxAnimator()
114{
115}
116
117void QskStackBoxAnimator::setStartIndex( int index )
118{
119 m_transientIndex = m_startIndex = index;
120}
121
122void QskStackBoxAnimator::setEndIndex( int index )
123{
124 m_endIndex = index;
125}
126
127int QskStackBoxAnimator::startIndex() const
128{
129 return m_startIndex;
130}
131
132int QskStackBoxAnimator::endIndex() const
133{
134 return m_endIndex;
135}
136
137QskStackBox* QskStackBoxAnimator::stackBox() const
138{
139 return static_cast< QskStackBox* >( parent() );
140}
141
142QQuickItem* QskStackBoxAnimator::itemAt( int index ) const
143{
144 return stackBox()->itemAtIndex(
145 ( index == 0 ) ? m_startIndex : m_endIndex );
146}
147
148qreal QskStackBoxAnimator::transientIndex() const
149{
150 return m_transientIndex;
151}
152
153void QskStackBoxAnimator::advance( qreal progress )
154{
155 qreal transientIndex;
156
157 if ( qFuzzyIsNull( progress ) )
158 {
159 transientIndex = m_startIndex;
160 }
161 else if ( qFuzzyCompare( progress, 1.0 ) )
162 {
163 transientIndex = m_endIndex;
164 }
165 else
166 {
167 const auto box = stackBox();
168
169 auto startIndex = m_startIndex;
170 auto endIndex = m_endIndex;
171
172 if ( startIndex == 0 )
173 {
174 if ( endIndex == box->itemCount() - 1 )
175 startIndex += box->itemCount();
176 }
177 else if ( startIndex == box->itemCount() - 1 )
178 {
179 if ( endIndex == 0 )
180 endIndex += box->itemCount();
181 }
182
183 transientIndex = startIndex + ( endIndex - startIndex ) * progress;
184 }
185
186 if ( !qskFuzzyCompare( m_transientIndex, transientIndex ) )
187 {
188 m_transientIndex = transientIndex;
189 advanceIndex( progress );
190
191 Q_EMIT stackBox()->transientIndexChanged( m_transientIndex );
192 }
193}
194
195QskStackBoxAnimator1::QskStackBoxAnimator1( QskStackBox* parent )
196 : QskStackBoxAnimator( parent )
197 , m_direction( Qsk::LeftToRight )
198 , m_isDirty( false )
199 , m_hasClip( false )
200{
201 // catching geometryChanges to know about resizing
202}
203
204QskStackBoxAnimator1::~QskStackBoxAnimator1()
205{
206}
207
208void QskStackBoxAnimator1::setDirection( Qsk::Direction direction )
209{
210 if ( m_direction != direction )
211 {
212 stop();
213 m_direction = direction;
214 }
215}
216
217Qsk::Direction QskStackBoxAnimator1::direction() const
218{
219 return m_direction;
220}
221
222void QskStackBoxAnimator1::setup()
223{
224 auto stackBox = this->stackBox();
225
226 m_hasClip = stackBox->clip();
227 if ( !m_hasClip )
228 stackBox->setClip( true );
229
230 stackBox->installEventFilter( this );
231 m_isDirty = true;
232}
233
234void QskStackBoxAnimator1::advanceIndex( qreal value )
235{
236 auto stackBox = this->stackBox();
237
238 const bool isHorizontal = ( m_direction == Qsk::LeftToRight )
239 || ( m_direction == Qsk::RightToLeft );
240
241 for ( int i = 0; i < 2; i++ )
242 {
243 if ( auto item = itemAt( i ) )
244 {
245 QRectF rect = qskItemGeometry( item );
246
247 if ( m_isDirty )
248 {
249 const int index = ( i == 0 ) ? startIndex() : endIndex();
250 rect = stackBox->geometryForItemAt( index );
251
252 m_itemOffset[ i ] = isHorizontal ? rect.x() : rect.y();
253 }
254
255 qreal x, y;
256
257 if ( isHorizontal )
258 {
259 qreal off = stackBox->width() * ( value - i );
260 if ( m_direction == Qsk::LeftToRight )
261 off = -off;
262
263 x = m_itemOffset[ i ] + off;
264 y = rect.y();
265 }
266 else
267 {
268 qreal off = stackBox->height() * ( value - i );
269 if ( m_direction == Qsk::BottomToTop )
270 off = -off;
271
272 x = rect.x();
273 y = m_itemOffset[ i ] + off;
274 }
275
276 qskSetItemGeometry( item, x, y, rect.width(), rect.height() );
277
278 if ( !item->isVisible() )
279 item->setVisible( true );
280 }
281 }
282
283 m_isDirty = false;
284}
285
286void QskStackBoxAnimator1::done()
287{
288 for ( int i = 0; i < 2; i++ )
289 {
290 if ( auto item = itemAt( i ) )
291 {
292 item->removeEventFilter( this );
293 item->setVisible( i == 1 );
294 }
295 }
296
297 if ( !m_hasClip )
298 stackBox()->setClip( false );
299}
300
301bool QskStackBoxAnimator1::eventFilter( QObject* object, QEvent* event )
302{
303 if ( !m_isDirty && object == stackBox() )
304 {
305 switch( static_cast< int >( event->type() ) )
306 {
307 case QskEvent::GeometryChange:
308 case QskEvent::ContentsRectChange:
309 case QskEvent::LayoutRequest:
310 {
311 m_isDirty = true;
312 break;
313 }
314 }
315 }
316
317 return QObject::eventFilter( object, event );
318}
319
320QskStackBoxAnimator2::QskStackBoxAnimator2( QskStackBox* parent )
321 : QskStackBoxAnimator( parent )
322 , m_orientation( Qt::Horizontal )
323 , m_inverted( false )
324{
325}
326
327QskStackBoxAnimator2::~QskStackBoxAnimator2()
328{
329}
330
331void QskStackBoxAnimator2::setOrientation( Qt::Orientation orientation )
332{
333 if ( m_orientation != orientation )
334 {
335 stop();
336 m_orientation = orientation;
337 }
338}
339
340Qt::Orientation QskStackBoxAnimator2::orientation() const
341{
342 return static_cast< Qt::Orientation >( m_orientation );
343}
344
345void QskStackBoxAnimator2::setInverted( bool on )
346{
347 if ( m_inverted != on )
348 {
349 stop();
350 m_inverted = on;
351 }
352}
353
354bool QskStackBoxAnimator2::isInverted() const
355{
356 return m_inverted;
357}
358
359void QskStackBoxAnimator2::setup()
360{
361 const auto axis = ( m_orientation == Qt::Horizontal )
362 ? Qt::YAxis : Qt::XAxis;
363
364 if ( auto item = itemAt( 0 ) )
365 ( void ) new RotationTransform( axis, 0.0, item );
366
367 if ( auto item = itemAt( 1 ) )
368 ( void ) new RotationTransform( axis, M_PI_2, item );
369}
370
371void QskStackBoxAnimator2::advanceIndex( qreal value )
372{
373 auto radians = value * M_PI;
374
375 if ( radians < M_PI_2 && radians > -M_PI_2 )
376 {
377 if ( auto item = itemAt( 0 ) )
378 {
379 if ( !m_inverted )
380 radians = 2 * M_PI - radians;
381
382 auto rotation = qskFindTransform< RotationTransform >( item );
383 rotation->setRadians( radians );
384
385 item->setVisible( true );
386 }
387
388 if ( auto item = itemAt( 1 ) )
389 {
390 item->setVisible( false );
391 }
392 }
393 else
394 {
395 if ( auto item = itemAt( 0 ) )
396 {
397 item->setVisible( false );
398 }
399
400 if ( auto item = itemAt( 1 ) )
401 {
402 radians = radians - M_PI;
403 if ( !m_inverted )
404 radians = 2 * M_PI - radians;
405
406 auto rotation = qskFindTransform< RotationTransform >( item );
407 rotation->setRadians( radians );
408
409 item->setVisible( true );
410 }
411 }
412}
413
414void QskStackBoxAnimator2::done()
415{
416 for ( int i = 0; i < 2; i++ )
417 {
418 if ( auto item = itemAt( i ) )
419 {
420 delete qskFindTransform< RotationTransform >( item );
421 item->setVisible( i == 1 );
422 }
423 }
424}
425
426QskStackBoxAnimator3::QskStackBoxAnimator3( QskStackBox* parent )
427 : QskStackBoxAnimator( parent )
428{
429}
430
431QskStackBoxAnimator3::~QskStackBoxAnimator3()
432{
433}
434
435void QskStackBoxAnimator3::setup()
436{
437 if ( auto item = itemAt( 1 ) )
438 {
439 item->setOpacity( 0.0 );
440 item->setVisible( true );
441 }
442}
443
444void QskStackBoxAnimator3::advanceIndex( qreal value )
445{
446 if ( auto item1 = itemAt( 0 ) )
447 item1->setOpacity( 1.0 - value );
448
449 if ( auto item2 = itemAt( 1 ) )
450 item2->setOpacity( value );
451}
452
453void QskStackBoxAnimator3::done()
454{
455 for ( int i = 0; i < 2; i++ )
456 {
457 if ( auto item = itemAt( i ) )
458 {
459 item->setOpacity( 1.0 );
460 item->setVisible( i == 1 ); // not here !!
461 }
462 }
463}
464
465QskStackBoxAnimator4::QskStackBoxAnimator4( QskStackBox* parent )
466 : QskStackBoxAnimator( parent )
467 , m_orientation( Qt::Horizontal )
468 , m_inverted( false )
469{
470}
471
472QskStackBoxAnimator4::~QskStackBoxAnimator4()
473{
474}
475
476void QskStackBoxAnimator4::setOrientation( Qt::Orientation orientation )
477{
478 if ( m_orientation != orientation )
479 {
480 stop();
481 m_orientation = orientation;
482 }
483}
484
485Qt::Orientation QskStackBoxAnimator4::orientation() const
486{
487 return m_orientation;
488}
489
490void QskStackBoxAnimator4::setInverted( bool on )
491{
492 if ( m_inverted != on )
493 {
494 stop();
495 m_inverted = on;
496 }
497}
498
499bool QskStackBoxAnimator4::isInverted() const
500{
501 return m_inverted;
502}
503
504void QskStackBoxAnimator4::setup()
505{
506 if ( auto item = itemAt( 0 ) )
507 {
508 ( void ) new QuickTransform( item );
509 item->setVisible( true );
510 }
511
512 if ( auto item = itemAt( 1 ) )
513 {
514 ( void ) new QuickTransform( item );
515 item->setVisible( true );
516 }
517}
518
519void QskStackBoxAnimator4::advanceIndex( qreal value )
520{
521 auto item1 = itemAt( 1 );
522 auto item2 = itemAt( 0 );
523
524 if ( isInverted() )
525 std::swap( item1, item2 );
526
527 const qreal posEdge = isInverted() ? value : 1.0 - value;
528
529 if ( item1 )
530 {
531 const auto transform = transformation( item1, false, posEdge );
532
533 auto rotation = qskFindTransform< QuickTransform >( item1 );
534 rotation->setTransform( transform );
535 }
536
537 if ( item2 )
538 {
539 const auto transform = transformation( item2, true, posEdge );
540
541 auto rotation = qskFindTransform< QuickTransform >( item2 );
542 rotation->setTransform( transform );
543 }
544}
545
546QTransform QskStackBoxAnimator4::transformation(
547 const QQuickItem* item, bool first, qreal posEdge ) const
548{
549 /*
550 first: left or top item
551 posEdge: position of the edge in the range of [0-1]
552 ( left->right, top->bottom ).
553 */
554
555 const qreal radians = M_PI_2 * ( 1.0 - posEdge );
556
557 QTransform transform;
558
559 if( orientation() == Qt::Horizontal )
560 {
561 const qreal dx = posEdge * ( item->x() + item->width() );
562 const qreal dy = 0.5 * item->height();
563
564 if ( first )
565 {
566 transform.translate( -item->x() + dx, dy );
567 transform.rotateRadians( radians, Qt::YAxis );
568 transform.translate( -item->width(), -dy );
569 }
570 else
571 {
572 transform.translate( dx, dy );
573 transform.rotateRadians( radians - M_PI_2, Qt::YAxis );
574 transform.translate( 0.0, -dy );
575 }
576 }
577 else
578 {
579 const qreal dx = 0.5 * item->width();
580 const qreal dy = posEdge * ( item->y() + item->height() );
581
582 if ( first )
583 {
584 transform.translate( dx, -item->y() + dy );
585 transform.rotateRadians( radians, Qt::XAxis );
586 transform.translate( -dx, -item->height() );
587 }
588 else
589 {
590 transform.translate( dx, dy );
591 transform.rotateRadians( radians - M_PI_2, Qt::XAxis );
592 transform.translate( -dx, 0.0 );
593 }
594 }
595
596 return transform;
597}
598
599void QskStackBoxAnimator4::done()
600{
601 for ( int i = 0; i < 2; i++ )
602 {
603 if ( auto item = itemAt( i ) )
604 {
605 delete qskFindTransform< QuickTransform >( item );
606 item->setVisible( i == 1 );
607 }
608 }
609}
610
611#include "moc_QskStackBoxAnimator.cpp"
612#include "QskStackBoxAnimator.moc"
Global definitions.
Direction
This enum type specifies a horizontal ot vertical direction.
@ LeftToRight
@ RightToLeft
@ BottomToTop