6#include "QskListViewSkinlet.h"
7#include "QskListView.h"
9#include "QskColorFilter.h"
10#include "QskGraphic.h"
11#include "QskBoxHints.h"
13#include "QskSkinStateChanger.h"
18#include <qtransform.h>
22 class ForegroundNode :
public QSGNode
27 removeAllChildNodes();
28 m_columnCount = m_oldRowMin = m_oldRowMax = -1;
31 void rearrangeNodes(
int rowMin,
int rowMax,
int columnCount )
33 const bool doReorder = ( columnCount == m_columnCount )
34 && ( rowMin <= m_oldRowMax ) && ( rowMax >= m_oldRowMin );
52 if ( rowMin >= m_oldRowMin )
54 for (
int row = m_oldRowMin; row < rowMin; row++ )
56 for (
int col = 0; col < columnCount; col++ )
58 auto childNode = firstChild();
59 removeChildNode( childNode );
60 appendChildNode( childNode );
66 for (
int row = rowMin; row < m_oldRowMin; row++ )
68 for (
int col = 0; col < columnCount; col++ )
70 auto childNode = lastChild();
71 removeChildNode( childNode );
72 prependChildNode( childNode );
80 m_columnCount = columnCount;
92 int m_columnCount = -1;
95 class ListViewNode final :
public QSGTransformNode
100 m_backgroundNode.setFlag( QSGNode::OwnedByParent,
false );
101 appendChildNode( &m_backgroundNode );
103 m_foregroundNode.setFlag( QSGNode::OwnedByParent,
false );
104 appendChildNode( &m_foregroundNode );
109 const auto scrollPos = listView->scrollPos();
110 setMatrix( QTransform::fromTranslate( -scrollPos.x(), -scrollPos.y() ) );
112 m_clipRect = listView->viewContentsRect();
113 m_rowHeight = listView->rowHeight();
115 m_rowMin = qFloor( scrollPos.y() / m_rowHeight );
117 const auto rowMax = ( scrollPos.y() + m_clipRect.height() ) / m_rowHeight;
118 m_rowMax = qFloor( rowMax - 10e-6 );
120 if ( m_rowMax >= listView->rowCount() )
121 m_rowMax = listView->rowCount() - 1;
124 QRectF clipRect()
const {
return m_clipRect; }
126 int rowMin()
const {
return m_rowMin; }
127 int rowMax()
const {
return m_rowMax; }
128 int rowCount()
const {
return m_rowMax - m_rowMin + 1; }
130 int rowHeight()
const {
return m_rowHeight; }
132 QSGNode* backgroundNode() {
return &m_backgroundNode; }
133 ForegroundNode* foregroundNode() {
return &m_foregroundNode; }
141 int m_rowMin, m_rowMax;
143 QSGNode m_backgroundNode;
144 ForegroundNode m_foregroundNode;
148static inline ListViewNode* qskListViewNode(
const QskListView* listView )
150 if (
auto node =
const_cast< QSGNode*
>( qskPaintNode( listView ) ) )
152 using namespace QskSGNode;
154 node = findChildNode( node, QskScrollViewSkinlet::ContentsRootRole );
157 node = node->firstChild();
160 Q_ASSERT( node->type() == QSGNode::TransformNodeType );
161 return static_cast< ListViewNode*
>( node );
169QskListViewSkinlet::QskListViewSkinlet(
QskSkin* skin )
174QskListViewSkinlet::~QskListViewSkinlet() =
default;
176QSGNode* QskListViewSkinlet::updateContentsNode(
179 const auto listView =
static_cast< const QskListView*
>( scrollView );
181 auto listViewNode = QskSGNode::ensureNode< ListViewNode >( node );
182 listViewNode->initialize( listView );
184 updateBackgroundNodes( listView, listViewNode->backgroundNode() );
185 updateForegroundNodes( listView, listViewNode->foregroundNode() );
190void QskListViewSkinlet::updateBackgroundNodes(
191 const QskListView* listView, QSGNode* backgroundNode )
const
195 auto listViewNode =
static_cast< const ListViewNode*
>( backgroundNode->parent() );
197 auto rowNode = backgroundNode->firstChild();
199 for (
int row = listViewNode->rowMin(); row <= listViewNode->rowMax(); row++ )
202 stateChanger.setStates( sampleStates( listView, Q::Cell, row ), row );
204 const auto rect = sampleRect( listView, listView->
contentsRect(), Q::Cell, row );
206 auto newNode = updateBoxNode( listView, rowNode, rect, Q::Cell );
209 if ( newNode->parent() != backgroundNode )
210 backgroundNode->appendChildNode( newNode );
212 rowNode = newNode->nextSibling();
216 QskSGNode::removeAllChildNodesFrom( backgroundNode, rowNode );
219void QskListViewSkinlet::updateForegroundNodes(
220 const QskListView* listView, QSGNode* parentNode )
const
222 auto foregroundNode =
static_cast< ForegroundNode*
>( parentNode );
224 if ( listView->rowCount() <= 0 || listView->columnCount() <= 0 )
226 foregroundNode->invalidate();
230 auto listViewNode =
static_cast< const ListViewNode*
>( parentNode->parent() );
232 const auto clipRect = listViewNode->clipRect();
234 const int rowMin = listViewNode->rowMin();
235 const int rowMax = listViewNode->rowMax();
237 foregroundNode->rearrangeNodes( rowMin, rowMax, listView->columnCount() );
241 const int colMin = 0;
242 const int colMax = listView->columnCount() - 1;
245 const auto margins = listView->
paddingHint( QskListView::Cell );
247 updateVisibleForegroundNodes(
248 listView, foregroundNode, rowMin, rowMax, margins );
251 auto node = foregroundNode->firstChild();
253 const auto rowHeight = listView->rowHeight();
254 auto y = clipRect.top() + rowMin * rowHeight;
256 for (
int row = rowMin; row <= rowMax; row++ )
258 qreal x = clipRect.left();
260 for (
int col = colMin; col <= colMax; col++ )
262 Q_ASSERT( node->type() == QSGNode::TransformNodeType );
263 auto transformNode =
static_cast< QSGTransformNode*
>( node );
265 transformNode->setMatrix(
266 QTransform::fromTranslate( x + margins.left(), y + margins.top() ) );
268 node = node->nextSibling();
269 x += listView->columnWidth( col );
276void QskListViewSkinlet::updateVisibleForegroundNodes(
278 int rowMin,
int rowMax,
const QMarginsF& margins )
const
280 auto node = parentNode->firstChild();
282 for (
int row = rowMin; row <= rowMax; row++ )
284 const auto h = listView->rowHeight() - ( margins.top() + margins.bottom() );
286 for (
int col = 0; col < listView->columnCount(); col++ )
288 const auto w = listView->columnWidth( col ) - ( margins.left() + margins.right() );
290 node = updateForegroundNode( listView,
291 parentNode,
static_cast< QSGTransformNode*
>( node ),
292 row, col, QSizeF( w, h ) );
294 node = node->nextSibling();
298 QskSGNode::removeAllChildNodesFrom( parentNode, node );
301QSGTransformNode* QskListViewSkinlet::updateForegroundNode(
302 const QskListView* listView, QSGNode* parentNode, QSGTransformNode* cellNode,
303 int row,
int col,
const QSizeF& size )
const
305 const QRectF cellRect( 0.0, 0.0, size.width(), size.height() );
311 QSGTransformNode* newCellNode =
nullptr;
313 if ( cellNode && ( cellNode->type() == QSGNode::TransformNodeType ) )
315 QSGNode* oldNode = cellNode;
317 auto newNode = updateCellNode( listView, oldNode, cellRect, row, col );
320 if ( newNode->type() == QSGNode::TransformNodeType )
322 newCellNode =
static_cast< QSGTransformNode*
>( newNode );
326 newCellNode =
new QSGTransformNode();
327 newCellNode->appendChildNode( newNode );
333 QSGNode* oldNode = cellNode ? cellNode->firstChild() :
nullptr;
334 auto newNode = updateCellNode( listView, oldNode, cellRect, row, col );
338 if ( newNode->type() == QSGNode::TransformNodeType )
340 newCellNode =
static_cast< QSGTransformNode*
>( newNode );
344 if ( cellNode ==
nullptr )
346 newCellNode =
new QSGTransformNode();
347 newCellNode->appendChildNode( newNode );
351 if ( newNode != oldNode )
353 delete cellNode->firstChild();
354 cellNode->appendChildNode( newNode );
356 newCellNode = cellNode;
363 if ( newCellNode ==
nullptr )
364 newCellNode =
new QSGTransformNode();
366 if ( cellNode != newCellNode )
370 parentNode->insertChildNodeAfter( newCellNode, cellNode );
375 parentNode->appendChildNode( newCellNode );
382QSGNode* QskListViewSkinlet::updateCellNode(
const QskListView* listView,
383 QSGNode* contentNode,
const QRectF& rect,
int row,
int col )
const
386 using namespace QskSGNode;
389 stateChanger.setStates( sampleStates( listView, Q::Cell, row ), row );
391 QSGNode* newNode =
nullptr;
401 Q::Cell, Qt::AlignVCenter | Qt::AlignLeft );
403 const auto value = listView->valueAt( row, col );
407 if ( nodeRole( contentNode ) == GraphicRole )
408 newNode = contentNode;
412 newNode = updateGraphicNode( listView, newNode,
413 value.value<
QskGraphic >(), colorFilter, rect, alignment );
416 setNodeRole( newNode, GraphicRole );
418 else if ( value.canConvert< QString >() )
420 if ( nodeRole( contentNode ) == TextRole )
421 newNode = contentNode;
423 newNode = updateTextNode( listView, newNode, rect, alignment,
424 value.toString(), Q::Text );
427 setNodeRole( newNode, TextRole );
431 qWarning() <<
"QskListViewSkinlet: got unsupported QVariant type" << value.typeName();
437QSizeF QskListViewSkinlet::sizeHint(
const QskSkinnable* skinnable,
438 Qt::SizeHint which,
const QSizeF& )
const
440 const auto listView =
static_cast< const QskListView*
>( skinnable );
444 if ( which != Qt::MaximumSize )
446 if ( listView->preferredWidthFromColumns() )
448 w = listView->scrollableSize().width();
449 w += listView->metric( QskScrollView::VerticalScrollBar |
QskAspect::Size );
453 return QSizeF( w, -1.0 );
456QskAspect::States QskListViewSkinlet::sampleStates(
const QskSkinnable* skinnable,
461 if ( subControl == Q::Cell || subControl == Q::Text || subControl == Q::Graphic )
463 const auto listView =
static_cast< const QskListView*
>( skinnable );
464 return listView->rowStates( index );
467 return Inherited::sampleStates( skinnable, subControl, index );
470QRectF QskListViewSkinlet::sampleRect(
const QskSkinnable* skinnable,
475 const auto listView =
static_cast< const QskListView*
>( skinnable );
477 if ( subControl == Q::Cell )
479 auto node = qskListViewNode( listView );
480 const auto clipRect = node ? node->clipRect() : listView->viewContentsRect();
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;
487 return QRectF( x, y, w, h );
490 return Inherited::sampleRect( skinnable, contentsRect, subControl, index );
493#include "moc_QskListViewSkinlet.cpp"
Subcontrol
For use within the rendering or lay-outing of a specific QskSkinnable.
QRectF contentsRect() const
A paint device for scalable graphics.
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.