10#include "QskArcMetrics.h"
11#include "QskBoxBorderColors.h"
12#include "QskBoxBorderMetrics.h"
13#include "QskBoxNode.h"
14#include "QskBoxRectangleNode.h"
15#include "QskBoxShapeMetrics.h"
16#include "QskBoxHints.h"
17#include "QskClipNode.h"
18#include "QskColorFilter.h"
19#include "QskControl.h"
20#include "QskFunctions.h"
21#include "QskGradient.h"
22#include "QskGraphicNode.h"
23#include "QskGraphic.h"
24#include "QskLinesNode.h"
26#include "QskStippleMetrics.h"
27#include "QskTextColors.h"
28#include "QskTextNode.h"
29#include "QskTextOptions.h"
30#include "QskSkinStateChanger.h"
31#include "QskTextureRenderer.h"
34#include <qquickwindow.h>
35#include <qsgsimplerectnode.h>
37static inline QRectF qskSceneAlignedRect(
const QQuickItem* item,
const QRectF& rect )
39 const auto transform = item->itemTransform(
nullptr,
nullptr );
40 if ( transform.type() > QTransform::TxTranslate )
50 const auto ratio = item->window()->devicePixelRatio();
52 const auto pos = transform.map( rect.topLeft() ) * ratio;
53 const auto size = rect.size() * ratio;
55 const qreal x = qRound( pos.x() ) / ratio;
56 const qreal y = qRound( pos.y() ) / ratio;
57 const qreal w = qRound( size.width() ) / ratio;
58 const qreal h = qRound( size.height() ) / ratio;
60 return QRectF( item->mapFromScene( QPointF( x, y ) ), QSizeF( w, h ) );
63static inline QRectF qskSubControlRect(
const QskSkinlet* skinlet,
68 const auto r = control->contentsRect();
69 return skinlet->subControlRect( skinnable, r, subControl );
75static inline QSGNode* qskUpdateTextNode(
const QskSkinnable* skinnable,
76 QSGNode* node,
const QRectF& rect, Qt::Alignment alignment,
77 const QString& text,
const QFont& font,
const QskTextOptions& textOptions,
80 if ( text.isEmpty() || rect.isEmpty() )
83 auto textNode =
static_cast< QskTextNode*
>( node );
84 if ( textNode ==
nullptr )
87 auto effectiveFont = font;
88 switch ( textOptions.fontSizeMode() )
90 case QskTextOptions::FixedSize:
93 case QskTextOptions::HorizontalFit:
97 case QskTextOptions::VerticalFit:
98 effectiveFont.setPixelSize(
static_cast< int >( rect.height() * 0.5 ) );
101 case QskTextOptions::Fit:
106 textNode->setTextData( skinnable->owningItem(),
107 text, rect, effectiveFont, textOptions, textColors, alignment, textStyle );
112static inline QSGNode* qskUpdateGraphicNode(
115 const QRectF& rect, Qt::Orientations mirrored )
117 if ( rect.isEmpty() )
120 const auto item = skinnable->owningItem();
121 if ( item ==
nullptr )
125 if ( graphicNode ==
nullptr )
130 bool useRaster = QskSetup::testUpdateFlag( flag );
131 if (
auto qItem = qobject_cast< const QskItem* >( item ) )
132 useRaster = qItem->testUpdateFlag( flag );
134 graphicNode->setRenderHint( useRaster ? QskPaintedNode::Raster :
QskPaintedNode::OpenGL );
136 graphicNode->setMirrored( mirrored );
138 const auto r = qskSceneAlignedRect( item, rect );
139 graphicNode->setGraphic( item->window(), graphic, colorFilter, r );
144static inline bool qskIsShadowVisible(
const QskShadowMetrics& shadowMetrics,
145 const QColor& shadowColor )
147 return !shadowMetrics.isNull() && shadowColor.isValid() && ( shadowColor.alpha() > 0 );
153 if ( gradient.isVisible() )
156 return !borderMetrics.isNull() && borderColors.isVisible();
159static inline bool qskIsArcVisible(
const QskArcMetrics& arcMetrics,
160 qreal borderWidth,
const QColor borderColor,
const QskGradient& gradient )
162 if ( arcMetrics.isNull() )
165 if ( borderWidth > 0.0 && borderColor.isValid() && borderColor.alpha() > 0 )
168 return gradient.isVisible();
171static inline bool qskIsLineVisible(
const QColor& lineColor, qreal lineWidth )
173 return ( lineWidth > 0.0 ) && lineColor.isValid() && ( lineColor.alpha() > 0 );
186 auto textColor = skinnable->
color( subControl, &status );
197static inline QQuickWindow* qskWindowOfSkinnable(
const QskSkinnable* skinnable )
199 if (
auto item = skinnable->owningItem() )
200 return item->window();
205static inline QSGNode* qskUpdateBoxNode(
206 const QskSkinnable* skinnable, QSGNode* node,
const QRectF& rect,
211 if ( !rect.isEmpty() )
213 if ( qskIsBoxVisible( borderMetrics, borderColors, gradient )
214 || qskIsShadowVisible( shadowMetrics, shadowColor ) )
216 if (
auto window = qskWindowOfSkinnable( skinnable ) )
218 auto boxNode = QskSGNode::ensureNode< QskBoxNode >( node );
219 boxNode->updateNode( window, rect, shape, borderMetrics,
220 borderColors, gradient, shadowMetrics, shadowColor );
230static inline QSGNode* qskUpdateArcNode(
232 qreal borderWidth,
const QColor borderColor,
235 if ( rect.isEmpty() )
238 if ( !qskIsArcVisible( metrics, borderWidth, borderColor, gradient ) )
241 auto arcNode = QskSGNode::ensureNode< QskArcNode >( node );
242 arcNode->setArcData( rect, metrics, borderWidth, borderColor, gradient );
247static inline QSGNode* qskUpdateLineNode(
248 const QskSkinnable*, QSGNode* node,
const QColor& lineColor,
254 if ( !qskIsLineVisible( lineColor, lineWidth ) )
257 auto linesNode = QskSGNode::ensureNode< QskLinesNode >( node );
258 linesNode->updateLine( lineColor, lineWidth, lineStipple,
259 QTransform(), line.p1(), line.p2() );
264static inline QSGNode* qskUpdateLinesNode(
265 const QskSkinnable*, QSGNode* node,
const QColor& lineColor,
266 qreal lineWidth,
QskStippleMetrics& lineStipple,
const QVector< QLineF >& lines )
268 if ( lines.isEmpty() )
271 if ( !qskIsLineVisible( lineColor, lineWidth ) )
274 auto linesNode = QskSGNode::ensureNode< QskLinesNode >( node );
275 linesNode->updateLines( lineColor, lineWidth, lineStipple,
276 QTransform(), lines );
281class QskSkinlet::PrivateData
286 , ownedBySkinnable( false )
291 QVector< quint8 > nodeRoles;
293 bool ownedBySkinnable : 1;
296QskSkinlet::QskSkinlet(
QskSkin* skin )
297 : m_data( new PrivateData( skin ) )
301QskSkinlet::~QskSkinlet()
305QskSkin* QskSkinlet::skin()
const
310void QskSkinlet::setOwnedBySkinnable(
bool on )
312 m_data->ownedBySkinnable = on;
315bool QskSkinlet::isOwnedBySkinnable()
const
317 return m_data->ownedBySkinnable;
320void QskSkinlet::setNodeRoles(
const QVector< quint8 >& nodeRoles )
322 m_data->nodeRoles = nodeRoles;
325void QskSkinlet::appendNodeRoles(
const QVector< quint8 >& nodeRoles )
327 m_data->nodeRoles += nodeRoles;
330const QVector< quint8 >& QskSkinlet::nodeRoles()
const
332 return m_data->nodeRoles;
335void QskSkinlet::updateNode(
QskSkinnable* skinnable, QSGNode* parentNode )
const
337 using namespace QskSGNode;
342 if (
const auto control = skinnable->
controlCast() )
346 oldNode = findChildNode( parentNode, BackgroundRole );
347 newNode = updateBackgroundNode( control, oldNode );
349 replaceChildNode( BackgroundRole, parentNode, oldNode, newNode );
353 oldNode = findChildNode( parentNode, DebugRole );
357 newNode = updateDebugNode( control, oldNode );
359 replaceChildNode( DebugRole, parentNode, oldNode, newNode );
362 for (
const auto nodeRole : std::as_const( m_data->nodeRoles ) )
364 Q_ASSERT( nodeRole < FirstReservedRole );
366 oldNode = QskSGNode::findChildNode( parentNode, nodeRole );
367 newNode = updateSubNode( skinnable, nodeRole, oldNode );
369 replaceChildNode( nodeRole, parentNode, oldNode, newNode );
373QSGNode* QskSkinlet::updateBackgroundNode(
374 const QskControl* control, QSGNode* node )
const
376 const auto rect = control->
rect();
377 if ( rect.isEmpty() )
381 if ( !gradient.isVisible() )
384 auto rectNode = QskSGNode::ensureNode< QskBoxRectangleNode >( node );
385 rectNode->updateFilling( control->window(), rect, gradient );
390QSGNode* QskSkinlet::updateDebugNode(
391 const QskControl* control, QSGNode* node )
const
393 if ( control->size().isEmpty() )
396 auto rectNode =
static_cast< QSGSimpleRectNode*
>( node );
397 if ( rectNode ==
nullptr )
399 rectNode =
new QSGSimpleRectNode();
402 if ( control->inherits(
"QskFocusIndicator" ) )
404 color = QColor( Qt::gray );
405 color.setAlpha( 60 );
410 idx = ( idx + 3 ) % 14;
413 color = QColor( Qt::GlobalColor( 4 + idx ) );
414 color.setAlpha( 200 );
417 rectNode->setColor( color );
420 const auto r = control->
rect();
421 if ( rectNode->rect() != r )
422 rectNode->setRect( r );
427void QskSkinlet::replaceChildNode( quint8 role,
428 QSGNode* parentNode, QSGNode* oldNode, QSGNode* newNode )
const
430 QskSGNode::replaceChildNode(
431 m_data->nodeRoles, role, parentNode, oldNode, newNode );
434QSGNode* QskSkinlet::updateBoxNode(
const QskSkinnable* skinnable,
437 const auto rect = qskSubControlRect(
this, skinnable, subControl );
438 return updateBoxNode( skinnable, node, rect, subControl );
441QSGNode* QskSkinlet::updateBoxNode(
const QskSkinnable* skinnable,
444 const auto fillGradient = skinnable->
gradientHint( subControl );
445 return updateBoxNode( skinnable, node, rect, fillGradient, subControl );
448QSGNode* QskSkinlet::updateBoxNode(
const QskSkinnable* skinnable,
449 QSGNode* node,
const QRectF& rect,
const QskGradient& fillGradient,
452 const auto margins = skinnable->
marginHint( subControl );
454 const auto boxRect = rect.marginsRemoved( margins );
455 if ( boxRect.isEmpty() )
460 const auto shape = skinnable->
boxShapeHint( subControl );
461 const auto shadowMetrics = skinnable->shadowMetricsHint( subControl );
462 const auto shadowColor = skinnable->shadowColorHint( subControl );
464 return qskUpdateBoxNode( skinnable, node,
465 boxRect, shape, borderMetrics, borderColors, fillGradient,
466 shadowMetrics, shadowColor );
469QSGNode* QskSkinlet::updateBoxNode(
470 const QskSkinnable* skinnable, QSGNode* node,
const QRectF& rect,
474 return qskUpdateBoxNode( skinnable, node,
475 rect, shape, borderMetrics, borderColors, fillGradient,
479QSGNode* QskSkinlet::updateBoxNode(
const QskSkinnable* skinnable,
480 QSGNode* node,
const QRectF& rect,
const QskBoxHints& hints )
482 return qskUpdateBoxNode( skinnable, node, rect,
483 hints.shape, hints.borderMetrics, hints.borderColors, hints.gradient,
484 hints.shadowMetrics, hints.shadowColor );
487QSGNode* QskSkinlet::updateInterpolatedBoxNode(
488 const QskSkinnable* skinnable, QSGNode* node,
const QRectF& rect,
494 ratio = qBound( 0.0, ratio, 1.0 );
496 if ( qFuzzyIsNull( ratio ) )
498 const auto margins = skinnable->
marginHint( aspect1 );
499 r = rect.marginsRemoved( margins );
501 boxHints = skinnable->boxHints( aspect1 ).toAbsolute( r.size() );
503 else if ( qFuzzyCompare( ratio, 1.0 ) )
505 const auto margins = skinnable->
marginHint( aspect2 );
506 r = rect.marginsRemoved( margins );
508 boxHints = skinnable->boxHints( aspect2 ).toAbsolute( r.size() );
513 margins = margins.interpolated( skinnable->
marginHint( aspect2 ), ratio );
515 r = rect.marginsRemoved( margins );
517 const auto boxHints1 = skinnable->boxHints( aspect1 ).toAbsolute( r.size() );
518 const auto boxHints2 = skinnable->boxHints( aspect2 ).toAbsolute( r.size() );
520 boxHints = boxHints1.interpolated( boxHints2, ratio );
523 return QskSkinlet::updateBoxNode( skinnable, node, r, boxHints );
526QSGNode* QskSkinlet::updateArcNode(
const QskSkinnable* skinnable,
529 const auto rect = qskSubControlRect(
this, skinnable, subControl );
530 return updateArcNode( skinnable, node, rect, subControl );
533QSGNode* QskSkinlet::updateArcNode(
const QskSkinnable* skinnable,
536 const auto fillGradient = skinnable->
gradientHint( subControl );
537 return updateArcNode( skinnable, node, rect, fillGradient, subControl );
540QSGNode* QskSkinlet::updateArcNode(
const QskSkinnable* skinnable,
541 QSGNode* node,
const QRectF& rect,
const QskGradient& fillGradient,
544 const auto metrics = skinnable->arcMetricsHint( subControl );
545 const auto r = rect.marginsRemoved( skinnable->
marginHint( subControl ) );
550 if ( borderWidth > 0.0 )
553 return qskUpdateArcNode( skinnable, node,
554 r, borderWidth, borderColor, fillGradient, metrics );
557QSGNode* QskSkinlet::updateArcNode(
558 const QskSkinnable* skinnable, QSGNode* node,
const QRectF& rect,
559 qreal borderWidth,
const QColor& borderColor,
562 return qskUpdateArcNode( skinnable, node, rect,
563 borderWidth, borderColor, fillGradient, metrics );
566QSGNode* QskSkinlet::updateArcNode(
const QskSkinnable* skinnable,
567 QSGNode* node, qreal startAngle, qreal spanAngle,
570 const auto rect = qskSubControlRect(
this, skinnable, subControl );
572 return updateArcNode( skinnable, node,
573 rect, startAngle, spanAngle, subControl );
576QSGNode* QskSkinlet::updateArcNode(
const QskSkinnable* skinnable,
577 QSGNode* node,
const QRectF& rect, qreal startAngle, qreal spanAngle,
580 const auto fillGradient = skinnable->
gradientHint( subControl );
581 return updateArcNode( skinnable, node, rect,
582 fillGradient, startAngle, spanAngle, subControl );
585QSGNode* QskSkinlet::updateArcNode(
const QskSkinnable* skinnable,
586 QSGNode* node,
const QRectF& rect,
const QskGradient& fillGradient,
589 auto arcMetrics = skinnable->arcMetricsHint( subControl );
590 arcMetrics.setStartAngle( startAngle );
591 arcMetrics.setSpanAngle( spanAngle );
596 if ( borderWidth > 0.0 )
599 const auto r = rect.marginsRemoved( skinnable->
marginHint( subControl ) );
600 return updateArcNode( skinnable, node, r,
601 borderWidth, borderColor, fillGradient, arcMetrics );
604QSGNode* QskSkinlet::updateLineNode(
const QskSkinnable* skinnable,
607 auto lineStipple = skinnable->stippleMetricsHint( subControl );
608 if ( !lineStipple.isValid() )
609 lineStipple = Qt::SolidLine;
611 const auto lineWidth = skinnable->metric( subControl |
QskAspect::Size );
612 const auto lineColor = skinnable->
color( subControl );
614 return qskUpdateLineNode( skinnable, node,
615 lineColor, lineWidth, lineStipple, line );
618QSGNode* QskSkinlet::updateLinesNode(
const QskSkinnable* skinnable,
621 auto lineStipple = skinnable->stippleMetricsHint( subControl );
622 if ( !lineStipple.isValid() )
623 lineStipple = Qt::SolidLine;
625 const auto lineWidth = skinnable->metric( subControl |
QskAspect::Size );
626 const auto lineColor = skinnable->
color( subControl );
628 return qskUpdateLinesNode( skinnable, node,
629 lineColor, lineWidth, lineStipple, lines );
632QSGNode* QskSkinlet::updateBoxClipNode(
const QskSkinnable* skinnable,
635 const auto rect = qskSubControlRect(
this, skinnable, subControl );
636 return updateBoxClipNode( skinnable, node, rect, subControl );
639QSGNode* QskSkinlet::updateBoxClipNode(
const QskSkinnable* skinnable,
642 auto clipNode = QskSGNode::ensureNode< QskClipNode >( node );
644 const auto margins = skinnable->
marginHint( subControl );
646 const auto clipRect = rect.marginsRemoved( margins );
647 if ( clipRect.isEmpty() )
649 clipNode->setRect( clipRect );
654 borderMetrics = borderMetrics.toAbsolute( clipRect.size() );
657 shape = shape.toAbsolute( clipRect.size() );
659 const auto window = qskWindowOfSkinnable( skinnable );
660 clipNode->setBox( window, clipRect, shape, borderMetrics );
666QSGNode* QskSkinlet::updateTextNode(
const QskSkinnable* skinnable,
667 QSGNode* node,
const QRectF& rect, Qt::Alignment alignment,
668 const QString& text,
const QFont& font,
const QskTextOptions& textOptions,
671 return qskUpdateTextNode( skinnable, node, rect, alignment,
672 text, font, textOptions, textColors, textStyle );
675QSGNode* QskSkinlet::updateTextNode(
677 const QRectF& rect, Qt::Alignment alignment,
680 const auto textOptions = skinnable->textOptionsHint( subControl );
682 return updateTextNode( skinnable, node, rect, alignment,
683 textOptions, text, subControl );
686QSGNode* QskSkinlet::updateTextNode(
const QskSkinnable* skinnable,
687 QSGNode* node,
const QRectF& rect,
691 if ( text.isEmpty() || rect.isEmpty() )
694 const auto colors = qskTextColors( skinnable, subControl );
696 auto style = Qsk::Normal;
697 if ( colors.styleColor().isValid() )
699 style = skinnable->
flagHint< Qsk::TextStyle >(
705 return qskUpdateTextNode( skinnable, node, rect, alignment,
706 text, font, textOptions, colors, style );
709QSGNode* QskSkinlet::updateTextNode(
713 const auto rect = qskSubControlRect(
this, skinnable, subControl );
714 const auto alignment = skinnable->
alignmentHint( subControl, Qt::AlignLeft );
716 return updateTextNode( skinnable, node,
717 rect, alignment, text, subControl );
720QSGNode* QskSkinlet::updateSymbolNode(
724 return updateGraphicNode( skinnable, node,
725 skinnable->symbolHint( subControl ), subControl );
728QSGNode* QskSkinlet::updateGraphicNode(
731 Qt::Orientations mirrored )
const
733 auto rect = qskSubControlRect(
this, skinnable, subControl );
734 rect = rect.marginsRemoved( skinnable->
marginHint( subControl ) );
736 const auto alignment = skinnable->
alignmentHint( subControl, Qt::AlignCenter );
739 return updateGraphicNode( skinnable, node,
740 graphic, colorFilter, rect, alignment, mirrored );
743QSGNode* QskSkinlet::updateGraphicNode(
746 const QRectF& rect, Qt::Alignment alignment, Qt::Orientations mirrored )
748 if ( graphic.isNull() )
751 const auto size = graphic.defaultSize().scaled(
752 rect.size(), Qt::KeepAspectRatio );
754 const auto r = qskAlignedRectF( rect, size, alignment );
755 return qskUpdateGraphicNode( skinnable, node, graphic, colorFilter, r, mirrored );
758QSGNode* QskSkinlet::updateGraphicNode(
761 const QRectF& rect, Qt::Orientations mirrored )
763 if ( graphic.isNull() )
766 return qskUpdateGraphicNode( skinnable, node, graphic, colorFilter, rect, mirrored );
769int QskSkinlet::sampleIndexAt(
const QskSkinnable* skinnable,
774 const auto count = sampleCount( skinnable, subControl );
776 for (
int i = 0; i < count; i++ )
778 const auto r = sampleRect( skinnable, rect, subControl, i );
779 if ( r.contains( pos ) )
786QSGNode* QskSkinlet::updateSeriesNode(
const QskSkinnable* skinnable,
789 auto node = rootNode ? rootNode->firstChild() :
nullptr;
790 QSGNode* lastNode =
nullptr;
792 const auto count = sampleCount( skinnable, subControl );
794 for(
int i = 0; i < count; i++ )
796 QSGNode* newNode =
nullptr;
799 const auto newStates = sampleStates( skinnable, subControl, i );
802 stateChanger.setStates( newStates, i );
804 newNode = updateSampleNode( skinnable, subControl, i, node );
809 if ( newNode == node )
811 node = node->nextSibling();
815 if ( rootNode ==
nullptr )
816 rootNode =
new QSGNode();
819 rootNode->insertChildNodeBefore( newNode, node );
821 rootNode->appendChildNode( newNode );
828 QskSGNode::removeAllChildNodesAfter( rootNode, lastNode );
833QSGNode* QskSkinlet::updateSampleNode(
const QskSkinnable*,
840QskAspect::States QskSkinlet::sampleStates(
844 return skinnable->skinStates();
854QSizeF QskSkinlet::hintWithoutConstraint(
855 const QSizeF& hint,
const QSizeF& constraint )
const
863 if ( constraint.width() < 0.0 )
864 h.setWidth( hint.width() );
866 if ( constraint.height() < 0.0 )
867 h.setHeight( hint.height() );
872#include "moc_QskSkinlet.cpp"
Lookup key for a QskSkinHintTable.
Subcontrol
For use within the rendering or lay-outing of a specific QskSkinnable.
Base class of all controls.
A paint device for scalable graphics.
@ PreferRasterForTextures
Describes the rendering interface of a QskControl. Change the skinlet to change the appearance of the...
QMarginsF marginHint(QskAspect, QskSkinHintStatus *=nullptr) const
Retrieves a margin hint.
QskControl * controlCast()
QskGradient gradientHint(QskAspect, QskSkinHintStatus *=nullptr) const
Retrieves a color hint as gradient.
QFont effectiveFont(QskAspect) const
QskBoxShapeMetrics boxShapeHint(QskAspect, QskSkinHintStatus *=nullptr) const
Retrieves a shape hint.
QskBoxBorderColors boxBorderColorsHint(QskAspect, QskSkinHintStatus *=nullptr) const
Retrieves border colors hint.
QskColorFilter effectiveGraphicFilter(QskAspect::Subcontrol) const
QColor color(QskAspect, QskSkinHintStatus *=nullptr) const
Retrieves a color hint.
QskBoxBorderMetrics boxBorderMetricsHint(QskAspect, QskSkinHintStatus *=nullptr) const
Retrieves a border hint.
T flagHint(QskAspect, T=T()) const
Retrieves a flag hint.
Qt::Alignment alignmentHint(QskAspect, Qt::Alignment=Qt::Alignment()) const
Retrieves an alignment hint.