QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskListViewSkinlet.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskListViewSkinlet.h"
7#include "QskListView.h"
8
9#include "QskColorFilter.h"
10#include "QskGraphic.h"
11#include "QskBoxHints.h"
12#include "QskSGNode.h"
13#include "QskSkinStateChanger.h"
14#include "QskQuick.h"
15
16#include <qmath.h>
17#include <qsgnode.h>
18#include <qtransform.h>
19
20namespace
21{
22 class ForegroundNode : public QSGNode
23 {
24 public:
25 void invalidate()
26 {
27 removeAllChildNodes();
28 m_columnCount = m_oldRowMin = m_oldRowMax = -1;
29 }
30
31 void rearrangeNodes( int rowMin, int rowMax, int columnCount )
32 {
33 const bool doReorder = ( columnCount == m_columnCount )
34 && ( rowMin <= m_oldRowMax ) && ( rowMax >= m_oldRowMin );
35
36 if ( doReorder )
37 {
38 /*
39 We have nodes that will be at a different position in the
40 list of child nodes - however their content might be unchanged.
41
42 This is a common situation, when resizing or scrolling the list.
43
44 To avoid that the text/graphic nodes have to be updated we
45 try to rearrange the nodes, so that they are in increasing
46 row order for further processing.
47
48 To avoid delete/new calls we append the leading and prepend
49 the trailing nodes, so that they might be used later.
50 */
51
52 if ( rowMin >= m_oldRowMin )
53 {
54 for ( int row = m_oldRowMin; row < rowMin; row++ )
55 {
56 for ( int col = 0; col < columnCount; col++ )
57 {
58 auto childNode = firstChild();
59 removeChildNode( childNode );
60 appendChildNode( childNode );
61 }
62 }
63 }
64 else
65 {
66 for ( int row = rowMin; row < m_oldRowMin; row++ )
67 {
68 for ( int col = 0; col < columnCount; col++ )
69 {
70 auto childNode = lastChild();
71 removeChildNode( childNode );
72 prependChildNode( childNode );
73 }
74 }
75 }
76 }
77
78 m_oldRowMin = rowMin;
79 m_oldRowMax = rowMax;
80 m_columnCount = columnCount;
81 }
82
83 private:
84 /*
85 When scrolling the majority of the child nodes are simply translated
86 while only few rows appear/disappear. To implement this in an efficient
87 way we store how the nodes were related to the rows in the previous update,
88 */
89
90 int m_oldRowMin = -1;
91 int m_oldRowMax = -1;
92 int m_columnCount = -1;
93 };
94
95 class ListViewNode final : public QSGTransformNode
96 {
97 public:
98 inline ListViewNode()
99 {
100 m_backgroundNode.setFlag( QSGNode::OwnedByParent, false );
101 appendChildNode( &m_backgroundNode );
102
103 m_foregroundNode.setFlag( QSGNode::OwnedByParent, false );
104 appendChildNode( &m_foregroundNode );
105 }
106
107 void initialize( const QskListView* listView )
108 {
109 const auto scrollPos = listView->scrollPos();
110 setMatrix( QTransform::fromTranslate( -scrollPos.x(), -scrollPos.y() ) );
111
112 m_clipRect = listView->viewContentsRect();
113 m_rowHeight = listView->rowHeight();
114
115 m_rowMin = qFloor( scrollPos.y() / m_rowHeight );
116
117 const auto rowMax = ( scrollPos.y() + m_clipRect.height() ) / m_rowHeight;
118 m_rowMax = qFloor( rowMax - 10e-6 );
119
120 if ( m_rowMax >= listView->rowCount() )
121 m_rowMax = listView->rowCount() - 1;
122 }
123
124 QRectF clipRect() const { return m_clipRect; }
125
126 int rowMin() const { return m_rowMin; }
127 int rowMax() const { return m_rowMax; }
128 int rowCount() const { return m_rowMax - m_rowMin + 1; }
129
130 int rowHeight() const { return m_rowHeight; }
131
132 QSGNode* backgroundNode() { return &m_backgroundNode; }
133 ForegroundNode* foregroundNode() { return &m_foregroundNode; }
134
135 private:
136 // caching some calculations to speed things up
137
138 QRectF m_clipRect;
139 qreal m_rowHeight;
140
141 int m_rowMin, m_rowMax;
142
143 QSGNode m_backgroundNode;
144 ForegroundNode m_foregroundNode;
145 };
146}
147
148static inline ListViewNode* qskListViewNode( const QskListView* listView )
149{
150 if ( auto node = const_cast< QSGNode* >( qskPaintNode( listView ) ) )
151 {
152 using namespace QskSGNode;
153
154 node = findChildNode( node, QskScrollViewSkinlet::ContentsRootRole );
155 if ( node )
156 {
157 node = node->firstChild();
158 if ( node )
159 {
160 Q_ASSERT( node->type() == QSGNode::TransformNodeType );
161 return static_cast< ListViewNode* >( node );
162 }
163 }
164 }
165
166 return nullptr;
167}
168
169QskListViewSkinlet::QskListViewSkinlet( QskSkin* skin )
170 : Inherited( skin )
171{
172}
173
174QskListViewSkinlet::~QskListViewSkinlet() = default;
175
176QSGNode* QskListViewSkinlet::updateContentsNode(
177 const QskScrollView* scrollView, QSGNode* node ) const
178{
179 const auto listView = static_cast< const QskListView* >( scrollView );
180
181 auto listViewNode = QskSGNode::ensureNode< ListViewNode >( node );
182 listViewNode->initialize( listView );
183
184 updateBackgroundNodes( listView, listViewNode->backgroundNode() );
185 updateForegroundNodes( listView, listViewNode->foregroundNode() );
186
187 return listViewNode;
188}
189
190void QskListViewSkinlet::updateBackgroundNodes(
191 const QskListView* listView, QSGNode* backgroundNode ) const
192{
193 using Q = QskListView;
194
195 auto listViewNode = static_cast< const ListViewNode* >( backgroundNode->parent() );
196
197 auto rowNode = backgroundNode->firstChild();
198
199 for ( int row = listViewNode->rowMin(); row <= listViewNode->rowMax(); row++ )
200 {
201 QskSkinStateChanger stateChanger( listView );
202 stateChanger.setStates( sampleStates( listView, Q::Cell, row ), row );
203
204 const auto rect = sampleRect( listView, listView->contentsRect(), Q::Cell, row );
205
206 auto newNode = updateBoxNode( listView, rowNode, rect, Q::Cell );
207 if ( newNode )
208 {
209 if ( newNode->parent() != backgroundNode )
210 backgroundNode->appendChildNode( newNode );
211 else
212 rowNode = newNode->nextSibling();
213 }
214 }
215
216 QskSGNode::removeAllChildNodesFrom( backgroundNode, rowNode );
217}
218
219void QskListViewSkinlet::updateForegroundNodes(
220 const QskListView* listView, QSGNode* parentNode ) const
221{
222 auto foregroundNode = static_cast< ForegroundNode* >( parentNode );
223
224 if ( listView->rowCount() <= 0 || listView->columnCount() <= 0 )
225 {
226 foregroundNode->invalidate();
227 return;
228 }
229
230 auto listViewNode = static_cast< const ListViewNode* >( parentNode->parent() );
231
232 const auto clipRect = listViewNode->clipRect();
233
234 const int rowMin = listViewNode->rowMin();
235 const int rowMax = listViewNode->rowMax();
236
237 foregroundNode->rearrangeNodes( rowMin, rowMax, listView->columnCount() );
238
239#if 1
240 // should be optimized for visible columns only
241 const int colMin = 0;
242 const int colMax = listView->columnCount() - 1;
243#endif
244
245 const auto margins = listView->paddingHint( QskListView::Cell );
246
247 updateVisibleForegroundNodes(
248 listView, foregroundNode, rowMin, rowMax, margins );
249
250 // finally putting the nodes into their position
251 auto node = foregroundNode->firstChild();
252
253 const auto rowHeight = listView->rowHeight();
254 auto y = clipRect.top() + rowMin * rowHeight;
255
256 for ( int row = rowMin; row <= rowMax; row++ )
257 {
258 qreal x = clipRect.left();
259
260 for ( int col = colMin; col <= colMax; col++ )
261 {
262 Q_ASSERT( node->type() == QSGNode::TransformNodeType );
263 auto transformNode = static_cast< QSGTransformNode* >( node );
264
265 transformNode->setMatrix(
266 QTransform::fromTranslate( x + margins.left(), y + margins.top() ) );
267
268 node = node->nextSibling();
269 x += listView->columnWidth( col );
270 }
271
272 y += rowHeight;
273 }
274}
275
276void QskListViewSkinlet::updateVisibleForegroundNodes(
277 const QskListView* listView, QSGNode* parentNode,
278 int rowMin, int rowMax, const QMarginsF& margins ) const
279{
280 auto node = parentNode->firstChild();
281
282 for ( int row = rowMin; row <= rowMax; row++ )
283 {
284 const auto h = listView->rowHeight() - ( margins.top() + margins.bottom() );
285
286 for ( int col = 0; col < listView->columnCount(); col++ )
287 {
288 const auto w = listView->columnWidth( col ) - ( margins.left() + margins.right() );
289
290 node = updateForegroundNode( listView,
291 parentNode, static_cast< QSGTransformNode* >( node ),
292 row, col, QSizeF( w, h ) );
293
294 node = node->nextSibling();
295 }
296 }
297
298 QskSGNode::removeAllChildNodesFrom( parentNode, node );
299}
300
301QSGTransformNode* QskListViewSkinlet::updateForegroundNode(
302 const QskListView* listView, QSGNode* parentNode, QSGTransformNode* cellNode,
303 int row, int col, const QSizeF& size ) const
304{
305 const QRectF cellRect( 0.0, 0.0, size.width(), size.height() );
306
307 /*
308 Text nodes already have a transform root node - to avoid inserting extra
309 transform nodes, the code below becomes a bit more complicated.
310 */
311 QSGTransformNode* newCellNode = nullptr;
312
313 if ( cellNode && ( cellNode->type() == QSGNode::TransformNodeType ) )
314 {
315 QSGNode* oldNode = cellNode;
316
317 auto newNode = updateCellNode( listView, oldNode, cellRect, row, col );
318 if ( newNode )
319 {
320 if ( newNode->type() == QSGNode::TransformNodeType )
321 {
322 newCellNode = static_cast< QSGTransformNode* >( newNode );
323 }
324 else
325 {
326 newCellNode = new QSGTransformNode();
327 newCellNode->appendChildNode( newNode );
328 }
329 }
330 }
331 else
332 {
333 QSGNode* oldNode = cellNode ? cellNode->firstChild() : nullptr;
334 auto newNode = updateCellNode( listView, oldNode, cellRect, row, col );
335
336 if ( newNode )
337 {
338 if ( newNode->type() == QSGNode::TransformNodeType )
339 {
340 newCellNode = static_cast< QSGTransformNode* >( newNode );
341 }
342 else
343 {
344 if ( cellNode == nullptr )
345 {
346 newCellNode = new QSGTransformNode();
347 newCellNode->appendChildNode( newNode );
348 }
349 else
350 {
351 if ( newNode != oldNode )
352 {
353 delete cellNode->firstChild();
354 cellNode->appendChildNode( newNode );
355
356 newCellNode = cellNode;
357 }
358 }
359 }
360 }
361 }
362
363 if ( newCellNode == nullptr )
364 newCellNode = new QSGTransformNode();
365
366 if ( cellNode != newCellNode )
367 {
368 if ( cellNode )
369 {
370 parentNode->insertChildNodeAfter( newCellNode, cellNode );
371 delete cellNode;
372 }
373 else
374 {
375 parentNode->appendChildNode( newCellNode );
376 }
377 }
378
379 return newCellNode;
380}
381
382QSGNode* QskListViewSkinlet::updateCellNode( const QskListView* listView,
383 QSGNode* contentNode, const QRectF& rect, int row, int col ) const
384{
385 using Q = QskListView;
386 using namespace QskSGNode;
387
388 QskSkinStateChanger stateChanger( listView );
389 stateChanger.setStates( sampleStates( listView, Q::Cell, row ), row );
390
391 QSGNode* newNode = nullptr;
392
393#if 1
394 /*
395 Alignments, text options etc are user definable attributes and need
396 to be adjustable - at least individually for each column - from the
397 public API of QskListView TODO ...
398 */
399#endif
400 const auto alignment = listView->alignmentHint(
401 Q::Cell, Qt::AlignVCenter | Qt::AlignLeft );
402
403 const auto value = listView->valueAt( row, col );
404
405 if ( value.canConvert< QskGraphic >() )
406 {
407 if ( nodeRole( contentNode ) == GraphicRole )
408 newNode = contentNode;
409
410 const auto colorFilter = listView->effectiveGraphicFilter( Q::Graphic );
411
412 newNode = updateGraphicNode( listView, newNode,
413 value.value< QskGraphic >(), colorFilter, rect, alignment );
414
415 if ( newNode )
416 setNodeRole( newNode, GraphicRole );
417 }
418 else if ( value.canConvert< QString >() )
419 {
420 if ( nodeRole( contentNode ) == TextRole )
421 newNode = contentNode;
422
423 newNode = updateTextNode( listView, newNode, rect, alignment,
424 value.toString(), Q::Text );
425
426 if ( newNode )
427 setNodeRole( newNode, TextRole );
428 }
429 else
430 {
431 qWarning() << "QskListViewSkinlet: got unsupported QVariant type" << value.typeName();
432 }
433
434 return newNode;
435}
436
437QSizeF QskListViewSkinlet::sizeHint( const QskSkinnable* skinnable,
438 Qt::SizeHint which, const QSizeF& ) const
439{
440 const auto listView = static_cast< const QskListView* >( skinnable );
441
442 qreal w = -1.0; // shouldn't we return something ???
443
444 if ( which != Qt::MaximumSize )
445 {
446 if ( listView->preferredWidthFromColumns() )
447 {
448 w = listView->scrollableSize().width();
449 w += listView->metric( QskScrollView::VerticalScrollBar | QskAspect::Size );
450 }
451 }
452
453 return QSizeF( w, -1.0 );
454}
455
456QskAspect::States QskListViewSkinlet::sampleStates( const QskSkinnable* skinnable,
457 QskAspect::Subcontrol subControl, int index ) const
458{
459 using Q = QskListView;
460
461 if ( subControl == Q::Cell || subControl == Q::Text || subControl == Q::Graphic )
462 {
463 const auto listView = static_cast< const QskListView* >( skinnable );
464 return listView->rowStates( index );
465 }
466
467 return Inherited::sampleStates( skinnable, subControl, index );
468}
469
470QRectF QskListViewSkinlet::sampleRect( const QskSkinnable* skinnable,
471 const QRectF& contentsRect, QskAspect::Subcontrol subControl, int index ) const
472{
473 using Q = QskListView;
474
475 const auto listView = static_cast< const QskListView* >( skinnable );
476
477 if ( subControl == Q::Cell )
478 {
479 auto node = qskListViewNode( listView );
480 const auto clipRect = node ? node->clipRect() : listView->viewContentsRect();
481
482 const auto w = clipRect.width();
483 const auto h = listView->rowHeight();
484 const auto x = clipRect.left() + listView->scrollPos().x();
485 const auto y = clipRect.top() + index * h;
486
487 return QRectF( x, y, w, h );
488 }
489
490 return Inherited::sampleRect( skinnable, contentsRect, subControl, index );
491}
492
493#include "moc_QskListViewSkinlet.cpp"
Subcontrol
For use within the rendering or lay-outing of a specific QskSkinnable.
Definition QskAspect.h:104
QRectF contentsRect() const
A paint device for scalable graphics.
Definition QskGraphic.h:28
QMarginsF paddingHint(QskAspect, QskSkinHintStatus *=nullptr) const
Retrieves a padding hint.
QskColorFilter effectiveGraphicFilter(QskAspect::Subcontrol) const
Qt::Alignment alignmentHint(QskAspect, Qt::Alignment=Qt::Alignment()) const
Retrieves an alignment hint.