QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskDirtyItemFilter.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskDirtyItemFilter.h"
7#include "QskItem.h"
8#include "QskInternalMacros.h"
9
10QSK_QT_PRIVATE_BEGIN
11#include <private/qquickitem_p.h>
12#include <private/qquickwindow_p.h>
13QSK_QT_PRIVATE_END
14
15static inline bool qskIsUpdateBlocked( const QQuickItem* item )
16{
17 if ( !item->isVisible() )
18 {
19 if ( auto qskItem = qobject_cast< const QskItem* >( item ) )
20 return qskItem->testUpdateFlag( QskItem::DeferredUpdate );
21 }
22
23#if 0
24 /*
25 Blocking items, that are outside the window would be easy,
26 but we have not yet found a performant way to send update notifications
27 when an item enters/leaves the window. TODO ...
28 */
29 else if ( const auto control = qskControlCast( item ) )
30 {
31 const QRectF itemRect( item->mapToScene( QPointF() ), item->size() );
32 const QRectF sceneRect( 0, 0, item->window()->width(), item->window()->height() );
33
34 return !itemRect.intersects( sceneRect );
35 }
36#endif
37
38 return false;
39}
40
41static inline void qskBlockDirty( QQuickItem* item, bool on )
42{
43 if ( qskIsUpdateBlocked( item ) )
44 QQuickItemPrivate::get( item )->componentComplete = !on;
45
46 const auto children = item->childItems();
47 for ( auto child : children )
48 qskBlockDirty( child, on );
49}
50
51namespace
52{
53 class ResetBlockedDirtyJob final : public QRunnable
54 {
55 public:
56 inline ResetBlockedDirtyJob( QQuickWindow* window )
57 : m_window( window )
58 {
59 }
60
61 void run() override
62 {
63 qskBlockDirty( m_window->contentItem(), false );
64 }
65
66 private:
67 const QQuickWindow* m_window;
68 };
69}
70
71QskDirtyItemFilter::QskDirtyItemFilter( QObject* parent )
72 : QObject( parent )
73{
74}
75
76QskDirtyItemFilter::~QskDirtyItemFilter()
77{
78}
79
80void QskDirtyItemFilter::addWindow( QQuickWindow* window )
81{
82 if ( m_windows.contains( window ) )
83 return;
84
85 m_windows.insert( window );
86
87 /*
88 Depending on the configration the scene graph runs on
89 a different thread and we need a direct connection to
90 filter the dirty list in the scene graph thread.
91
92 As QObject::sender() is not valid, when using a direct
93 connection between different threads, we need an
94 extra lambda to pass window as parameter.
95 */
96
97 connect( window, &QQuickWindow::beforeSynchronizing,
98 window, [ this, window ] { beforeSynchronizing( window ); },
99 Qt::DirectConnection );
100
101 connect( window, &QObject::destroyed,
102 this, [ this, window ] { m_windows.remove( window ); } );
103}
104
105void QskDirtyItemFilter::beforeSynchronizing( QQuickWindow* window )
106{
107 filterDirtyList( window, qskIsUpdateBlocked );
108
109 if ( QQuickWindowPrivate::get( window )->renderer == nullptr )
110 {
111 /*
112 In this specific initial situation QQuickWindow updates
113 the dirtyList after emitting "beforeSynchronizing".
114 So we need a special hack setting/resetting componentComplete
115 to avoid having all items in the dirtyList.
116 */
117 qskBlockDirty( window->contentItem(), true );
118
119 window->scheduleRenderJob( new ResetBlockedDirtyJob( window ),
120 QQuickWindow::AfterSynchronizingStage );
121 }
122}
123
124void QskDirtyItemFilter::filterDirtyList(
125 QQuickWindow* window, bool ( *isBlocked )( const QQuickItem* ) )
126{
127 if ( window == nullptr )
128 return;
129
130 auto d = QQuickWindowPrivate::get( window );
131 for ( auto item = d->dirtyItemList; item != nullptr; )
132 {
133 auto nextItem = QQuickItemPrivate::get( item )->nextDirtyItem;
134
135 if ( isBlocked( item ) )
136 QQuickItemPrivate::get( item )->removeFromDirtyList();
137
138 item = nextItem;
139 }
140}
@ DeferredUpdate
Definition QskItem.h:52