QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskItem.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskItem.h"
7#include "QskItemPrivate.h"
8#include "QskQuick.h"
9#include "QskEvent.h"
10#include "QskSetup.h"
11#include "QskSkinManager.h"
12#include "QskSkin.h"
13#include "QskDirtyItemFilter.h"
14#include "QskInternalMacros.h"
15
16#include <qglobalstatic.h>
17#include <qquickwindow.h>
18
19#if defined( QT_DEBUG )
20
21QSK_QT_PRIVATE_BEGIN
22
23#if QT_VERSION >= QT_VERSION_CHECK( 6, 2, 0 )
24 #ifndef emit
25 #define emit
26 #include <private/qabstractanimation_p.h>
27 #undef emit
28 #endif
29#endif
30
31#if defined( QT_DEBUG )
32#include <private/qquickpositioners_p.h>
33#endif
34
35#include <private/qquickitemchangelistener_p.h>
36
37QSK_QT_PRIVATE_END
38
39#endif
40
41#include <unordered_set>
42
43static inline void qskSendEventTo( QObject* object, QEvent::Type type )
44{
45 QEvent event( type );
46 QCoreApplication::sendEvent( object, &event );
47}
48
49static inline void qskApplyUpdateFlags(
50 QskItem::UpdateFlags flags, QskItem* item )
51{
52 auto d = static_cast< QskItemPrivate* >( QskItemPrivate::get( item ) );
53 d->applyUpdateFlags( flags );
54}
55
56static inline void qskFilterWindow( QQuickWindow* window )
57{
58 if ( window == nullptr )
59 return;
60
61 static QskDirtyItemFilter itemFilter;
62 itemFilter.addWindow( window );
63}
64
65namespace
66{
67 class QskItemRegistry
68 {
69 public:
70 QskItemRegistry()
71 {
72 /*
73 Its faster and saves some memory to have this registry instead
74 of setting up direct connections between qskSetup and each control
75
76 We would also need to send QEvent::StyleChange, when
77 a window has a new skin. TODO ...
78 */
79 QObject::connect( qskSkinManager, &QskSkinManager::skinChanged,
80 qskSkinManager, [ this ] { updateSkin(); } );
81
82 QObject::connect( qskSkinManager, &QskSkinManager::colorSchemeChanged,
83 qskSkinManager, [ this ] { updateSkin(); } );
84 }
85
86 inline void insert( QskItem* item )
87 {
88 m_items.insert( item );
89 }
90
91 inline void remove( QskItem* item )
92 {
93 m_items.erase( item );
94 }
95
96 void updateItemFlags()
97 {
98 const auto flags = QskSetup::updateFlags();
99
100 for ( auto item : m_items )
101 qskApplyUpdateFlags( flags, item );
102 }
103
104 void updateSkin()
105 {
106 QEvent event( QEvent::StyleChange );
107
108 for ( auto item : m_items )
109 {
110 event.setAccepted( true );
111 QCoreApplication::sendEvent( item, &event );
112 }
113 }
114
115 private:
116 std::unordered_set< QskItem* > m_items;
117 };
118}
119
120namespace
121{
122 /*
123 A helper class to store the released window to be able to
124 put it later into the WindowChange event.
125 */
126 class QskWindowStore
127 {
128 public:
129 QskWindowStore()
130 : m_refCount( 0 )
131 , m_window( nullptr )
132 {
133 }
134
135 void setWindow( QQuickWindow* window )
136 {
137 if ( m_window != window )
138 {
139 m_window = window;
140 m_refCount = 0;
141 }
142
143 if ( m_window )
144 m_refCount++;
145 }
146
147 QQuickWindow* window()
148 {
149 QQuickWindow* w = m_window;
150
151 if ( m_window )
152 {
153 if ( --m_refCount == 0 )
154 m_window = nullptr;
155 }
156
157 return w;
158 }
159
160 private:
161 int m_refCount;
162 QQuickWindow* m_window;
163 };
164}
165
166namespace
167{
168 // A helper class for the polishOnParentResize feature
169
170 class QskParentListener final : public QQuickItemChangeListener
171 {
172 public:
173 void update( QQuickItem* parentItem )
174 {
175 if ( parentItem == nullptr )
176 return;
177
178 const auto changeTypes =
179 QQuickItemPrivate::Geometry | QQuickItemPrivate::Children;
180
181 auto d = QQuickItemPrivate::get( parentItem );
182
183 if ( needListening( parentItem ) )
184 d->updateOrAddItemChangeListener( this, changeTypes );
185 else
186 d->removeItemChangeListener( this, changeTypes );
187 }
188
189 private:
190 inline bool needListening( const QQuickItem* parentItem ) const
191 {
192 const auto children = parentItem->childItems();
193 for ( auto child : children )
194 {
195 if ( auto item = qobject_cast< const QskItem* >( child ) )
196 {
197 if ( item->polishOnParentResize() )
198 return true;
199 }
200 }
201
202 return false;
203 }
204
205 void itemGeometryChanged( QQuickItem* parentItem,
206 QQuickGeometryChange, const QRectF& ) override
207 {
208 const auto children = parentItem->childItems();
209 for ( auto child : children )
210 {
211 if ( auto item = qobject_cast< QskItem* >( child ) )
212 {
213 if ( item->polishOnParentResize() )
214 {
215 item->resetImplicitSize();
216 item->polish();
217 }
218 }
219 }
220 }
221
222 void itemChildRemoved(
223 QQuickItem* parentItem, QQuickItem* ) override
224 {
225 update( parentItem );
226 }
227 };
228}
229
230Q_GLOBAL_STATIC( QskItemRegistry, qskRegistry )
231Q_GLOBAL_STATIC( QskWindowStore, qskReleasedWindowCounter )
232Q_GLOBAL_STATIC( QskParentListener, qskParentListener )
233
234QskItem::QskItem( QskItemPrivate& dd, QQuickItem* parent )
235 : QQuickItem( dd, parent )
236{
237 setFlag( QQuickItem::ItemHasContents, true );
238 Inherited::setActiveFocusOnTab( false );
239
240 if ( dd.updateFlags & QskItem::DeferredUpdate )
241 qskFilterWindow( window() );
242
243 qskRegistry->insert( this );
244}
245
246QskItem::~QskItem()
247{
248 /*
249 We set componentComplete to false, so that operations
250 that are triggered by detaching the item from its parent
251 can be aware of the about-to-delete state.
252
253 Note, that since Qt >= 6.5 this information is stored
254 in QQuickItemPrivate::inDestructor.
255
256 s.a: qskIsItemInDestructor
257 */
258 d_func()->componentComplete = false;
259
260 if ( qskRegistry )
261 qskRegistry->remove( this );
262}
263
264const char* QskItem::className() const
265{
266 return metaObject()->className();
267}
268
270{
271 Inherited::classBegin();
272}
273
275{
276#if defined( QT_DEBUG )
277 if ( d_func()->updateFlags & QskItem::DeferredLayout )
278 {
279 if ( qobject_cast< const QQuickBasePositioner* >( parent() ) )
280 {
281 qWarning( "QskItem in DeferredLayout mode under control of a positioner" );
282 }
283 }
284#endif
285
286 Inherited::componentComplete();
287}
288
290{
291 Inherited::releaseResources();
292
293 // QQuickItem::derefWindow runs over the children between
294 // calling releaseResources and itemChange. So we need to have
295 // a reference count to know, when we have processed all
296 // sequences to be able to provide the correct "oldWindow"
297 // in the WindowChange event.
298
299 qskReleasedWindowCounter->setWindow( window() );
300}
301
302void QskItem::setDisabled( bool on )
303{
304 Inherited::setEnabled( !on );
305}
306
307void QskItem::setHidden( bool on )
308{
309 Inherited::setVisible( !on );
310}
311
313{
314 Inherited::setVisible( true );
315}
316
318{
319 Inherited::setVisible( false );
320}
321
322bool QskItem::isVisibleTo( const QQuickItem* ancestor ) const
323{
324 return qskIsVisibleTo( this, ancestor );
325}
326
328{
329 return d_func()->explicitVisible;
330}
331
332void QskItem::setGeometry( qreal x, qreal y, qreal width, qreal height )
333{
334 // QQuickItem does not even offer changing the geometry
335 // in one call - what leads to 2 calls of the updateGeometry
336 // hook. Grmpf ...
337
338 Q_D( QQuickItem );
339
340#if QT_VERSION >= QT_VERSION_CHECK( 6, 2, 0 )
341 d->heightValidFlag = true;
342 d->widthValidFlag = true;
343#else
344 d->heightValid = true;
345 d->widthValid = true;
346#endif
347
348 const QRectF oldRect( d->x, d->y, d->width, d->height );
349
350 int dirtyType = 0;
351
352 if ( d->x != x || d->y != y )
353 {
354 d->x = x;
355 d->y = y;
356
357 dirtyType |= QQuickItemPrivate::Position;
358 }
359
360 if ( d->width != width || d->height != height )
361 {
362 d->height = height;
363 d->width = width;
364
365 dirtyType |= QQuickItemPrivate::Size;
366 }
367
368 if ( dirtyType )
369 {
370 if ( dirtyType & QQuickItemPrivate::Position )
371 d->dirty( QQuickItemPrivate::Position );
372
373 if ( dirtyType & QQuickItemPrivate::Size )
374 d->dirty( QQuickItemPrivate::Size );
375
376 /*
377 Unfortunately geometryChange(d) is protected and we can't implement
378 this code as qskSetItemGeometry - further hacking required: TODO ...
379 */
380
381 const QRectF newRect( d->x, d->y, d->width, d->height );
382#if QT_VERSION >= QT_VERSION_CHECK( 6, 0, 0 )
383 geometryChange( newRect, oldRect );
384#else
385 geometryChanged( newRect, oldRect );
386#endif
387 }
388}
389
390QRectF QskItem::rect() const
391{
392 Q_D( const QQuickItem );
393 return QRectF( 0, 0, d->width, d->height );
394}
395
396QRectF QskItem::geometry() const
397{
398 Q_D( const QQuickItem );
399 return QRectF( d->x, d->y, d->width, d->height );
400}
401
402void QskItem::setTabFence( bool on )
403{
404 Q_D( QQuickItem );
405 if ( on != d->isTabFence )
406 {
407 d->isTabFence = on;
408 Q_EMIT itemFlagsChanged();
409 }
410}
411
413{
414 return d_func()->isTabFence;
415}
416
417#if QT_VERSION < QT_VERSION_CHECK( 6, 7, 0 )
418
419void QskItem::setFocusPolicy( Qt::FocusPolicy policy )
420{
421 Q_D( QskItem );
422 if ( policy != d->focusPolicy )
423 {
424 d->focusPolicy = ( policy & ~Qt::TabFocus );
425
426 const bool tabFocus = policy & Qt::TabFocus;
427
428 if ( !tabFocus && window() )
429 {
430 // removing the activeFocusItem from the focus tab chain is not possible
431 if ( window()->activeFocusItem() == this )
432 {
433 if ( auto focusItem = nextItemInFocusChain( true ) )
434 focusItem->setFocus( true );
435 }
436 }
437
438 Inherited::setActiveFocusOnTab( tabFocus );
439
440 Q_EMIT focusPolicyChanged( focusPolicy() );
441 }
442}
443
444Qt::FocusPolicy QskItem::focusPolicy() const
445{
446 uint policy = d_func()->focusPolicy;
447 if ( Inherited::activeFocusOnTab() )
448 policy |= Qt::TabFocus;
449
450 return static_cast< Qt::FocusPolicy >( policy );
451}
452
453#else
454
455/*
456 shifting the activeFocusItem before removing it from the tab focus
457 chain is not done by QQuickItem::setFocusPolicy. TODO ...
458 */
459#endif
460
461void QskItem::setWheelEnabled( bool on )
462{
463 Q_D( QskItem );
464 if ( on != d->wheelEnabled )
465 {
466 d->wheelEnabled = on;
467 Q_EMIT wheelEnabledChanged( on );
468 }
469}
470
471bool QskItem::isWheelEnabled() const
472{
473 return d_func()->wheelEnabled;
474}
475
477{
478 Q_D( QskItem );
479 if ( on != d->polishOnResize )
480 {
481 d->polishOnResize = on;
482 polish();
483
484 Q_EMIT itemFlagsChanged();
485 }
486}
487
489{
490 return d_func()->polishOnResize;
491}
492
493void QskItem::setPolishOnParentResize( bool on )
494{
495 Q_D( QskItem );
496 if ( on != d->polishOnParentResize )
497 {
498 d->polishOnParentResize = on;
499
500 if ( parentItem() && qskParentListener )
501 {
502 qskParentListener->update( parentItem() );
503
505 polish();
506 }
507
508 Q_EMIT itemFlagsChanged();
509 }
510}
511
512bool QskItem::polishOnParentResize() const
513{
514 return d_func()->polishOnParentResize;
515}
516
518{
519#if 1
520 /*
521 What about using Qt::LayoutDirection instead. It sounds
522 like a more expressive API and we do not run into conflicts
523 with the layoutMirroring() attached property for QML.
524 But what is the situation with locales, where the default direction
525 is RightToLeft ?
526 */
527 return d_func()->effectiveLayoutMirror;
528#endif
529}
530
531void QskItem::setLayoutMirroring( bool on, bool childrenInherit )
532{
533 // Again we have to deal with an existing API made for QML,
534 // that is weired for C++: LayoutMirroring/QQuickLayoutMirroringAttached
535 // Internally it is managed by 5(!) different flags - condolences
536 // to the poor guy who has been sentenced to maintain this.
537
538 // Anyway, the code below might achieve the desired behavior without
539 // breaking the QML path.
540
541 Q_D( QQuickItem );
542
543 if ( childrenInherit != d->inheritMirrorFromItem )
544 {
545 d->inheritMirrorFromItem = childrenInherit;
546 d->resolveLayoutMirror();
547 }
548
549 d->isMirrorImplicit = false;
550
551 if ( on != d->effectiveLayoutMirror )
552 {
553 d->setLayoutMirror( on );
554 if ( childrenInherit )
555 d->resolveLayoutMirror();
556 }
557}
558
560{
561 Q_D( QQuickItem );
562
563 if ( d && !d->isMirrorImplicit )
564 {
565 d->isMirrorImplicit = true;
566 // d->inheritMirrorFromItem = false;
567 d->resolveLayoutMirror();
568 }
569}
570
572{
573 return d_func()->polishScheduled;
574}
575
577{
578 Q_D( const QskItem );
579
580 return ( d->dirtyAttributes & QQuickItemPrivate::ContentUpdateMask ) &&
581 ( d->flags & QQuickItem::ItemHasContents );
582}
583
585{
586 return d_func()->initiallyPainted;
587}
588
590{
591 Q_D( const QskItem );
592
593 if ( d->width <= 0.0 && d->height <= 0.0 )
594 {
595 /*
596 Unfortunately the list of items to-be-polished is not processed
597 in top/down order and we might run into updatePolish() before
598 having a proper size. But when the parentItem() is waiting
599 for to-be-polished, we assume, that we will be resized then
600 and run into another updatePolish() then.
601 */
602 if ( d->polishOnResize && qskIsPolishScheduled( parentItem() ) )
603 return true;
604 }
605
606 return false;
607}
608
609QskItem::UpdateFlags QskItem::updateFlags() const
610{
611 return UpdateFlags( d_func()->updateFlags );
612}
613
615{
616 Q_D( QskItem );
617
618 // clear all bits in the mask
619 d->updateFlagsMask = 0;
620 d->applyUpdateFlags( QskSetup::updateFlags() );
621}
622
624{
625 Q_D( QskItem );
626
627 d->updateFlagsMask |= flag;
628
629 if ( testUpdateFlag( flag ) != on )
630 {
631 applyUpdateFlag( flag, on );
633 }
634}
635
637{
638 Q_D( QskItem );
639
640 d->updateFlagsMask &= ~flag;
641
642 const bool on = QskSetup::testUpdateFlag( flag );
643
644 if ( testUpdateFlag( flag ) != on )
645 {
646 applyUpdateFlag( flag, on );
648 }
649}
650
652{
653 return d_func()->updateFlags & flag;
654}
655
656void QskItem::applyUpdateFlag( UpdateFlag flag, bool on )
657{
658 Q_D( QskItem );
659
660 if ( testUpdateFlag( flag ) == on )
661 return;
662
663 if ( on )
664 d->updateFlags |= flag;
665 else
666 d->updateFlags &= ~flag;
667
668 switch ( flag )
669 {
671 {
672 if ( on )
673 {
674 qskFilterWindow( window() );
675 }
676 else
677 {
678 if ( !isVisible() )
679 update();
680 }
681
682 break;
683 }
685 {
686 if ( !on && d->blockedPolish )
687 polish();
688
689 break;
690 }
692 {
693 if ( !on )
694 {
695 // Update the implicitSize and rebind the size to it.
696 // Having set the size explicitly gets lost.
697
698#if QT_VERSION >= QT_VERSION_CHECK( 6, 2, 0 )
699 d->widthValidFlag = d->heightValidFlag = false;
700#else
701 d->widthValid = d->heightValid = false;
702#endif
703 d->updateImplicitSize( false );
704 }
705
706 break;
707 }
709 {
710 if ( on && !isVisible() )
711 d->cleanupNodes();
712
713 break;
714 }
716 {
717 // no need to mark it dirty
718 if ( flags() & QQuickItem::ItemHasContents )
719 update();
720 break;
721 }
722 default:
723 break;
724 }
725}
726
728{
729 Q_D( QskItem );
730
731 if ( d->updateFlags & QskItem::DeferredLayout )
732 {
733 d->blockedImplicitSize = true;
734 d->layoutConstraintChanged();
735 }
736 else
737 {
738 d->updateImplicitSize( true );
739 }
740}
741
742bool QskItem::event( QEvent* event )
743{
744 const int eventType = event->type();
745 const bool hasContents = flags() & QQuickItem::ItemHasContents;
746
747 switch( eventType )
748 {
749 case QEvent::StyleChange:
750 {
751 d_func()->clearPreviousNodes = true;
752
754 polish();
755
756 if ( hasContents )
757 update();
758
760 return true;
761 }
762 case QEvent::ContentsRectChange:
763 {
765
766 if ( d_func()->polishOnResize )
767 polish();
768
769 if ( hasContents )
770 update();
771
773 return true;
774 }
775 case QEvent::ReadOnlyChange:
776 case QEvent::EnabledChange:
777 case QEvent::LocaleChange:
778 case QEvent::ParentChange:
779 case QEvent::LayoutDirectionChange:
780 {
782 return true;
783 }
784 case QskEvent::ViewportChange:
785 {
786 viewportChangeEvent( static_cast< QskViewportChangeEvent* >( event ) );
787 return true;
788 }
789 case QskEvent::GeometryChange:
790 {
792 return true;
793 }
794 case QskEvent::WindowChange:
795 {
796 windowChangeEvent( static_cast< QskWindowChangeEvent* >( event ) );
797 return true;
798 }
799 case QEvent::LayoutRequest:
800 {
801 if ( d_func()->polishOnResize )
802 polish();
803
804 return true;
805 }
806 case QEvent::FocusIn:
807 {
808 if ( window() == nullptr )
809 {
810 /*
811 During deconstruction of the window we run into
812 focus changes when the items in the tree get destroyed.
813 Calling focusInEvent() in this state does not make sense
814 and often results in crashes in overloaded event handlers.
815 */
816 return true;
817 }
818
819 break;
820 }
821 case QEvent::MouseButtonPress:
822 case QEvent::MouseButtonRelease:
823 {
824 if ( ( focusPolicy() & Qt::ClickFocus ) == Qt::ClickFocus )
825 {
826 if ( QGuiApplication::styleHints()->setFocusOnTouchRelease() )
827 {
828 if ( event->type() == QEvent::MouseButtonRelease )
829 forceActiveFocus( Qt::MouseFocusReason );
830 }
831 else
832 {
833 if ( event->type() == QEvent::MouseButtonPress )
834 forceActiveFocus( Qt::MouseFocusReason );
835 }
836 }
837 break;
838 }
839 case QEvent::Wheel:
840 {
841 if ( !isWheelEnabled() )
842 {
843 /*
844 We block further processing of the event. This is in line
845 with not receiving any mouse event that have not been
846 explicitly enabled with setAcceptedMouseButtons().
847 */
848 event->ignore();
849 return true;
850 }
851
852 if ( ( focusPolicy() & Qt::WheelFocus ) == Qt::WheelFocus )
853 forceActiveFocus( Qt::MouseFocusReason );
854
855 break;
856 }
857 }
858
859 return Inherited::event( event );
860}
861
865
866void QskItem::viewportChangeEvent( QskViewportChangeEvent* event )
867{
868 event->ignore();
869}
870
874
875void QskItem::changeEvent( QEvent* )
876{
877}
878
879void QskItem::itemChange( QQuickItem::ItemChange change,
880 const QQuickItem::ItemChangeData& changeData )
881{
882 switch ( change )
883 {
884 case QQuickItem::ItemSceneChange:
885 {
886 if ( changeData.window )
887 {
888 Q_D( const QskItem );
889 if ( d->updateFlags & QskItem::DeferredUpdate )
890 qskFilterWindow( changeData.window );
891 }
892
893 auto oldWindow = qskReleasedWindowCounter->window();
894
895 if ( oldWindow && oldWindow->contentItem()
896 && ( oldWindow->activeFocusItem() == this ) )
897 {
898 /*
899 Removing an item from the scene might result in
900 changes of the active focus item. Unfortunately the corresponding
901 FocusIn/Out events are sent, while the item tree is in an
902 invalid state.
903 When having event handlers, that do modifications of the focus
904 ( f.e. assigning the local focus, inside of a focus scope )
905 we might end up with having a dangling pointer for
906 oldWindow->activeFocusItem().
907 */
908
909#if QT_VERSION >= QT_VERSION_CHECK( 6, 1, 0 )
910 auto wd = QQuickWindowPrivate::get( oldWindow )->deliveryAgentPrivate();
911#else
912 auto wd = QQuickWindowPrivate::get( oldWindow );
913#endif
914 if ( auto scope = qskNearestFocusScope( this ) )
915 {
916 wd->clearFocusInScope( scope, this, Qt::OtherFocusReason );
917 }
918 else
919 {
920 wd->activeFocusItem = nullptr;
921 }
922 }
923 if ( changeData.window == nullptr )
924 {
925 Q_D( QskItem );
926
927 if( d->focus )
928 {
929 /*
930 The focus flag is not cleared, when removing an
931 item from the window. In situations where the item gets
932 reinserted into the window - or transferred to another one -
933 we might run into situations, where 2 items in the same scope
934 have the "focus" flag being set.
935 A better solution might be to check the flag when reinserting
936 into a window ...
937 */
938 d->focus = false;
939 }
940 }
941
942 QskWindowChangeEvent event( oldWindow, changeData.window );
943 QCoreApplication::sendEvent( this, &event );
944
945 break;
946 }
947 case QQuickItem::ItemEnabledHasChanged:
948 {
949 qskSendEventTo( this, QEvent::EnabledChange );
950 break;
951 }
952 case QQuickItem::ItemVisibleHasChanged:
953 {
954 Q_D( QskItem );
955
956 if ( changeData.boolValue )
957 {
958 if ( d->blockedPolish )
959 polish();
960
961 if ( d->updateFlags & QskItem::DeferredUpdate )
962 {
963 if ( d->dirtyAttributes && ( d->flags & QQuickItem::ItemHasContents ) )
964 update();
965 }
966 }
967 else
968 {
969 if ( d->updateFlags & QskItem::CleanupOnVisibility )
970 d->cleanupNodes();
971
972 d->initiallyPainted = false;
973 }
974
975 if ( parentItem() && parentItem()->isVisible() )
976 {
977 /*
978 Layout code might consider the visiblility of the children
979 and therefore needs to be updated. Posting a statement about
980 changed layout constraints has this effect, but is not correct.
981 The right way to go would be to create show/hide events and to
982 handle them, where visibility of the children matters.
983 TODO ...
984 */
985
986 d->layoutConstraintChanged();
987 }
988
989 break;
990 }
991 case QQuickItem::ItemParentHasChanged:
992 {
993 if( polishOnParentResize() && qskParentListener )
994 qskParentListener->update( parentItem() );
995
996 break;
997 }
998 case QQuickItem::ItemChildAddedChange:
999 case QQuickItem::ItemChildRemovedChange:
1000 {
1001 // do we want to have events for those ???
1002 break;
1003 }
1004
1005 case QQuickItem::ItemOpacityHasChanged:
1006 case QQuickItem::ItemActiveFocusHasChanged:
1007 case QQuickItem::ItemRotationHasChanged:
1008 case QQuickItem::ItemAntialiasingHasChanged:
1009 case QQuickItem::ItemDevicePixelRatioHasChanged:
1010#if QT_VERSION >= QT_VERSION_CHECK( 6, 9, 0 )
1011 case QQuickItem::ItemScaleHasChanged:
1012 case QQuickItem::ItemTransformHasChanged:
1013#endif
1014 {
1015 break;
1016 }
1017 }
1018
1019 Inherited::itemChange( change, changeData );
1020}
1021
1022#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
1023
1024void QskItem::geometryChanged(
1025 const QRectF& newGeometry, const QRectF& oldGeometry )
1026{
1027 geometryChange( newGeometry, oldGeometry );
1028}
1029
1030#endif
1031
1033 const QRectF& newGeometry, const QRectF& oldGeometry )
1034{
1035#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
1036 Inherited::geometryChanged( newGeometry, oldGeometry );
1037#else
1038 Inherited::geometryChange( newGeometry, oldGeometry );
1039#endif
1040
1041 Q_D( const QskItem );
1042 if ( !d->polishScheduled && d->polishOnResize )
1043 {
1044 if ( newGeometry.size() != oldGeometry.size() )
1045 polish();
1046 }
1047
1048 QskGeometryChangeEvent event( newGeometry, oldGeometry );
1049 QCoreApplication::sendEvent( this, &event );
1050}
1051
1053{
1054 Inherited::mouseUngrabEvent();
1055}
1056
1058{
1059 Inherited::touchUngrabEvent();
1060}
1061
1062#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
1063
1064void QskItem::windowDeactivateEvent()
1065{
1066 Inherited::windowDeactivateEvent();
1067}
1068
1069#endif
1070
1071void QskItem::updatePolish()
1072{
1073 Q_D( QskItem );
1074
1075 if ( d->updateFlags & QskItem::DeferredPolish )
1076 {
1077 if ( !isVisible() )
1078 {
1079 d->blockedPolish = true;
1080 return;
1081 }
1082 }
1083
1084 d->blockedPolish = false;
1085
1086 if ( !d->initiallyPainted )
1087 {
1088 /*
1089 We should find a better way for identifying, when
1090 an item is about to be shown, than making it dependend
1091 from polishing and the existence of scene graph nodes. TODO ...
1092 */
1093 aboutToShow();
1094 }
1095
1096 updateItemPolish();
1097}
1098
1100{
1101}
1102
1103void QskItem::updateItemPolish()
1104{
1105}
1106
1107QSGNode* QskItem::updatePaintNode( QSGNode* node, UpdatePaintNodeData* data )
1108{
1109 Q_UNUSED( data )
1110
1111 Q_D( QskItem );
1112
1113 Q_ASSERT( isVisible() || !( d->updateFlags & QskItem::DeferredUpdate ) );
1114
1115 d->initiallyPainted = true;
1116
1117 if ( d->clearPreviousNodes )
1118 {
1119 delete node;
1120 node = nullptr;
1121#if 1
1122 /*
1123 controls might find subnodes using qskPaintNode - not good
1124 as d->paintNode is not updated before leaving here. TODO ...
1125
1126 In the initial call we will always have a nullptr - even if
1127 it has already been allocated. When deleting it we have a dangling pointer.
1128 instead of the new one.
1129
1130 To avoid creashes for the second situation we manually clear d->paintNode.
1131 */
1132 d->paintNode = nullptr;
1133#endif
1134
1135 d->clearPreviousNodes = false;
1136 }
1137
1138 return updateItemPaintNode( node );
1139}
1140
1141QSGNode* QskItem::updateItemPaintNode( QSGNode* node )
1142{
1143 return node;
1144}
1145
1146QSK_HIDDEN_EXTERNAL_BEGIN
1147
1148void qskUpdateItemFlags()
1149{
1150 if ( qskRegistry )
1151 qskRegistry->updateItemFlags();
1152}
1153
1154QSK_HIDDEN_EXTERNAL_END
1155
1156#include "moc_QskItem.cpp"
bool isPolishScheduled() const
Definition QskItem.cpp:571
void mouseUngrabEvent() override
Definition QskItem.cpp:1052
void setTabFence(bool)
Definition QskItem.cpp:402
void show()
Definition QskItem.cpp:312
bool polishOnResize
Definition QskItem.h:28
bool isVisibleToParent() const
Definition QskItem.cpp:327
void geometryChange(const QRectF &, const QRectF &) override
Definition QskItem.cpp:1032
void setHidden(bool)
Definition QskItem.cpp:307
void resetUpdateFlags()
Definition QskItem.cpp:614
void itemChange(ItemChange, const ItemChangeData &) override
Definition QskItem.cpp:879
bool event(QEvent *) override
Definition QskItem.cpp:742
UpdateFlags updateFlags
Definition QskItem.h:45
void setPolishOnResize(bool)
Definition QskItem.cpp:476
void classBegin() override
Definition QskItem.cpp:269
QRectF rect
Definition QskItem.h:22
void resetImplicitSize()
Definition QskItem.cpp:727
virtual void aboutToShow()
Definition QskItem.cpp:1099
void resetLayoutMirroring()
Definition QskItem.cpp:559
void touchUngrabEvent() override
Definition QskItem.cpp:1057
void releaseResources() override
Definition QskItem.cpp:289
const char * className() const
Definition QskItem.cpp:264
void setDisabled(bool)
Definition QskItem.cpp:302
UpdateFlag
Definition QskItem.h:51
@ DebugForceBackground
Definition QskItem.h:59
@ CleanupOnVisibility
Definition QskItem.h:55
@ DeferredPolish
Definition QskItem.h:53
@ DeferredUpdate
Definition QskItem.h:52
@ DeferredLayout
Definition QskItem.h:54
void itemFlagsChanged()
void setLayoutMirroring(bool on, bool childrenInherit=false)
Definition QskItem.cpp:531
bool isTabFence() const
Definition QskItem.cpp:412
virtual void geometryChangeEvent(QskGeometryChangeEvent *)
Definition QskItem.cpp:871
bool isInitiallyPainted() const
Definition QskItem.cpp:584
void resetUpdateFlag(UpdateFlag)
Definition QskItem.cpp:636
virtual void changeEvent(QEvent *)
Definition QskItem.cpp:875
virtual void windowChangeEvent(QskWindowChangeEvent *)
Definition QskItem.cpp:862
bool isUpdateNodeScheduled() const
Definition QskItem.cpp:576
QRectF geometry
Definition QskItem.h:21
void updateFlagsChanged(UpdateFlags)
bool maybeUnresized() const
Definition QskItem.cpp:589
void setUpdateFlag(UpdateFlag, bool on=true)
Definition QskItem.cpp:623
void hide()
Definition QskItem.cpp:317
bool isVisibleTo(const QQuickItem *) const
Definition QskItem.cpp:322
bool testUpdateFlag(UpdateFlag) const
Definition QskItem.cpp:651
void componentComplete() override
Definition QskItem.cpp:274
void setGeometry(qreal x, qreal y, qreal width, qreal height)
Definition QskItem.cpp:332
bool layoutMirroring() const
Definition QskItem.cpp:517