QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskSkinlet.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskSkinlet.h"
7
8#include "QskArcNode.h"
9#include "QskAspect.h"
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"
25#include "QskSGNode.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"
32#include "QskSetup.h"
33
34#include <qquickwindow.h>
35#include <qsgsimplerectnode.h>
36
37static inline QRectF qskSceneAlignedRect( const QQuickItem* item, const QRectF& rect )
38{
39 const auto transform = item->itemTransform( nullptr, nullptr );
40 if ( transform.type() > QTransform::TxTranslate )
41 return rect;
42
43 /*
44 Aligning rect according to scene coordinates, so that
45 we don't run into rounding issues downstream, where values
46 will be floored/ceiled ending up with a slightly different
47 aspect ratio.
48 */
49
50 const auto ratio = item->window()->devicePixelRatio();
51
52 const auto pos = transform.map( rect.topLeft() ) * ratio;
53 const auto size = rect.size() * ratio;
54
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;
59
60 return QRectF( item->mapFromScene( QPointF( x, y ) ), QSizeF( w, h ) );
61}
62
63static inline QRectF qskSubControlRect( const QskSkinlet* skinlet,
64 const QskSkinnable* skinnable, QskAspect::Subcontrol subControl )
65{
66 if ( auto control = skinnable->controlCast() )
67 {
68 const auto r = control->contentsRect();
69 return skinlet->subControlRect( skinnable, r, subControl );
70 }
71
72 return QRectF();
73}
74
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,
78 const QskTextColors& textColors, Qsk::TextStyle textStyle )
79{
80 if ( text.isEmpty() || rect.isEmpty() )
81 return nullptr;
82
83 auto textNode = static_cast< QskTextNode* >( node );
84 if ( textNode == nullptr )
85 textNode = new QskTextNode();
86
87 auto effectiveFont = font;
88 switch ( textOptions.fontSizeMode() )
89 {
90 case QskTextOptions::FixedSize:
91 break;
92
93 case QskTextOptions::HorizontalFit:
94 Q_UNIMPLEMENTED();
95 break;
96
97 case QskTextOptions::VerticalFit:
98 effectiveFont.setPixelSize( static_cast< int >( rect.height() * 0.5 ) );
99 break;
100
101 case QskTextOptions::Fit:
102 Q_UNIMPLEMENTED();
103 break;
104 }
105
106 textNode->setTextData( skinnable->owningItem(),
107 text, rect, effectiveFont, textOptions, textColors, alignment, textStyle );
108
109 return textNode;
110}
111
112static inline QSGNode* qskUpdateGraphicNode(
113 const QskSkinnable* skinnable, QSGNode* node,
114 const QskGraphic& graphic, const QskColorFilter& colorFilter,
115 const QRectF& rect, Qt::Orientations mirrored )
116{
117 if ( rect.isEmpty() )
118 return nullptr;
119
120 const auto item = skinnable->owningItem();
121 if ( item == nullptr )
122 return nullptr;
123
124 auto graphicNode = static_cast< QskGraphicNode* >( node );
125 if ( graphicNode == nullptr )
126 graphicNode = new QskGraphicNode();
127
128 const auto flag = QskItem::PreferRasterForTextures;
129
130 bool useRaster = QskSetup::testUpdateFlag( flag );
131 if ( auto qItem = qobject_cast< const QskItem* >( item ) )
132 useRaster = qItem->testUpdateFlag( flag );
133
134 graphicNode->setRenderHint( useRaster ? QskPaintedNode::Raster : QskPaintedNode::OpenGL );
135
136 graphicNode->setMirrored( mirrored );
137
138 const auto r = qskSceneAlignedRect( item, rect );
139 graphicNode->setGraphic( item->window(), graphic, colorFilter, r );
140
141 return graphicNode;
142}
143
144static inline bool qskIsShadowVisible( const QskShadowMetrics& shadowMetrics,
145 const QColor& shadowColor )
146{
147 return !shadowMetrics.isNull() && shadowColor.isValid() && ( shadowColor.alpha() > 0 );
148}
149
150static inline bool qskIsBoxVisible( const QskBoxBorderMetrics& borderMetrics,
151 const QskBoxBorderColors& borderColors, const QskGradient& gradient )
152{
153 if ( gradient.isVisible() )
154 return true;
155
156 return !borderMetrics.isNull() && borderColors.isVisible();
157}
158
159static inline bool qskIsArcVisible( const QskArcMetrics& arcMetrics,
160 qreal borderWidth, const QColor borderColor, const QskGradient& gradient )
161{
162 if ( arcMetrics.isNull() )
163 return false;
164
165 if ( borderWidth > 0.0 && borderColor.isValid() && borderColor.alpha() > 0 )
166 return true;
167
168 return gradient.isVisible();
169}
170
171static inline bool qskIsLineVisible( const QColor& lineColor, qreal lineWidth )
172{
173 return ( lineWidth > 0.0 ) && lineColor.isValid() && ( lineColor.alpha() > 0 );
174}
175
176static inline QskTextColors qskTextColors(
177 const QskSkinnable* skinnable, QskAspect::Subcontrol subControl )
178{
179 /*
180 Would be more efficient to have QskTextColors hints instead of
181 storing the colors as seperated hints. TODO ...
182 */
183
184 QskSkinHintStatus status;
185
186 auto textColor = skinnable->color( subControl, &status );
187#if 1
188 if ( !status.isValid() )
189 textColor = skinnable->color( subControl | QskAspect::TextColor );
190#endif
191
192 return QskTextColors( textColor,
193 skinnable->color( subControl | QskAspect::StyleColor ),
194 skinnable->color( subControl | QskAspect::LinkColor ) );
195}
196
197static inline QQuickWindow* qskWindowOfSkinnable( const QskSkinnable* skinnable )
198{
199 if ( auto item = skinnable->owningItem() )
200 return item->window();
201
202 return nullptr;
203}
204
205static inline QSGNode* qskUpdateBoxNode(
206 const QskSkinnable* skinnable, QSGNode* node, const QRectF& rect,
207 const QskBoxShapeMetrics& shape, const QskBoxBorderMetrics& borderMetrics,
208 const QskBoxBorderColors& borderColors, const QskGradient& gradient,
209 const QskShadowMetrics& shadowMetrics, const QColor& shadowColor )
210{
211 if ( !rect.isEmpty() )
212 {
213 if ( qskIsBoxVisible( borderMetrics, borderColors, gradient )
214 || qskIsShadowVisible( shadowMetrics, shadowColor ) )
215 {
216 if ( auto window = qskWindowOfSkinnable( skinnable ) )
217 {
218 auto boxNode = QskSGNode::ensureNode< QskBoxNode >( node );
219 boxNode->updateNode( window, rect, shape, borderMetrics,
220 borderColors, gradient, shadowMetrics, shadowColor );
221
222 return boxNode;
223 }
224 }
225 }
226
227 return nullptr;
228}
229
230static inline QSGNode* qskUpdateArcNode(
231 const QskSkinnable*, QSGNode* node, const QRectF& rect,
232 qreal borderWidth, const QColor borderColor,
233 const QskGradient& gradient, const QskArcMetrics& metrics )
234{
235 if ( rect.isEmpty() )
236 return nullptr;
237
238 if ( !qskIsArcVisible( metrics, borderWidth, borderColor, gradient ) )
239 return nullptr;
240
241 auto arcNode = QskSGNode::ensureNode< QskArcNode >( node );
242 arcNode->setArcData( rect, metrics, borderWidth, borderColor, gradient );
243
244 return arcNode;
245}
246
247static inline QSGNode* qskUpdateLineNode(
248 const QskSkinnable*, QSGNode* node, const QColor& lineColor,
249 qreal lineWidth, QskStippleMetrics& lineStipple, const QLineF& line )
250{
251 if ( line.isNull() )
252 return nullptr;
253
254 if ( !qskIsLineVisible( lineColor, lineWidth ) )
255 return nullptr;
256
257 auto linesNode = QskSGNode::ensureNode< QskLinesNode >( node );
258 linesNode->updateLine( lineColor, lineWidth, lineStipple,
259 QTransform(), line.p1(), line.p2() );
260
261 return linesNode;
262}
263
264static inline QSGNode* qskUpdateLinesNode(
265 const QskSkinnable*, QSGNode* node, const QColor& lineColor,
266 qreal lineWidth, QskStippleMetrics& lineStipple, const QVector< QLineF >& lines )
267{
268 if ( lines.isEmpty() )
269 return nullptr;
270
271 if ( !qskIsLineVisible( lineColor, lineWidth ) )
272 return nullptr;
273
274 auto linesNode = QskSGNode::ensureNode< QskLinesNode >( node );
275 linesNode->updateLines( lineColor, lineWidth, lineStipple,
276 QTransform(), lines );
277
278 return linesNode;
279}
280
281class QskSkinlet::PrivateData
282{
283 public:
284 PrivateData( QskSkin* skin )
285 : skin( skin )
286 , ownedBySkinnable( false )
287 {
288 }
289
290 QskSkin* skin;
291 QVector< quint8 > nodeRoles;
292
293 bool ownedBySkinnable : 1;
294};
295
296QskSkinlet::QskSkinlet( QskSkin* skin )
297 : m_data( new PrivateData( skin ) )
298{
299}
300
301QskSkinlet::~QskSkinlet()
302{
303}
304
305QskSkin* QskSkinlet::skin() const
306{
307 return m_data->skin;
308}
309
310void QskSkinlet::setOwnedBySkinnable( bool on )
311{
312 m_data->ownedBySkinnable = on;
313}
314
315bool QskSkinlet::isOwnedBySkinnable() const
316{
317 return m_data->ownedBySkinnable;
318}
319
320void QskSkinlet::setNodeRoles( const QVector< quint8 >& nodeRoles )
321{
322 m_data->nodeRoles = nodeRoles;
323}
324
325void QskSkinlet::appendNodeRoles( const QVector< quint8 >& nodeRoles )
326{
327 m_data->nodeRoles += nodeRoles;
328}
329
330const QVector< quint8 >& QskSkinlet::nodeRoles() const
331{
332 return m_data->nodeRoles;
333}
334
335void QskSkinlet::updateNode( QskSkinnable* skinnable, QSGNode* parentNode ) const
336{
337 using namespace QskSGNode;
338
339 QSGNode* oldNode;
340 QSGNode* newNode;
341
342 if ( const auto control = skinnable->controlCast() )
343 {
344 // background
345
346 oldNode = findChildNode( parentNode, BackgroundRole );
347 newNode = updateBackgroundNode( control, oldNode );
348
349 replaceChildNode( BackgroundRole, parentNode, oldNode, newNode );
350
351 // debug
352
353 oldNode = findChildNode( parentNode, DebugRole );
354
355 newNode = nullptr;
356 if ( control->testUpdateFlag( QskItem::DebugForceBackground ) )
357 newNode = updateDebugNode( control, oldNode );
358
359 replaceChildNode( DebugRole, parentNode, oldNode, newNode );
360 }
361
362 for ( const auto nodeRole : std::as_const( m_data->nodeRoles ) )
363 {
364 Q_ASSERT( nodeRole < FirstReservedRole );
365
366 oldNode = QskSGNode::findChildNode( parentNode, nodeRole );
367 newNode = updateSubNode( skinnable, nodeRole, oldNode );
368
369 replaceChildNode( nodeRole, parentNode, oldNode, newNode );
370 }
371}
372
373QSGNode* QskSkinlet::updateBackgroundNode(
374 const QskControl* control, QSGNode* node ) const
375{
376 const auto rect = control->rect();
377 if ( rect.isEmpty() )
378 return nullptr;
379
380 const auto gradient = control->background();
381 if ( !gradient.isVisible() )
382 return nullptr;
383
384 auto rectNode = QskSGNode::ensureNode< QskBoxRectangleNode >( node );
385 rectNode->updateFilling( control->window(), rect, gradient );
386
387 return rectNode;
388}
389
390QSGNode* QskSkinlet::updateDebugNode(
391 const QskControl* control, QSGNode* node ) const
392{
393 if ( control->size().isEmpty() )
394 return nullptr;
395
396 auto rectNode = static_cast< QSGSimpleRectNode* >( node );
397 if ( rectNode == nullptr )
398 {
399 rectNode = new QSGSimpleRectNode();
400
401 QColor color;
402 if ( control->inherits( "QskFocusIndicator" ) )
403 {
404 color = QColor( Qt::gray );
405 color.setAlpha( 60 );
406 }
407 else
408 {
409 static int idx = 0;
410 idx = ( idx + 3 ) % 14;
411
412 // maybe using something random based on web colors ???
413 color = QColor( Qt::GlobalColor( 4 + idx ) );
414 color.setAlpha( 200 );
415 }
416
417 rectNode->setColor( color );
418 }
419
420 const auto r = control->rect();
421 if ( rectNode->rect() != r )
422 rectNode->setRect( r );
423
424 return rectNode;
425}
426
427void QskSkinlet::replaceChildNode( quint8 role,
428 QSGNode* parentNode, QSGNode* oldNode, QSGNode* newNode ) const
429{
430 QskSGNode::replaceChildNode(
431 m_data->nodeRoles, role, parentNode, oldNode, newNode );
432}
433
434QSGNode* QskSkinlet::updateBoxNode( const QskSkinnable* skinnable,
435 QSGNode* node, QskAspect::Subcontrol subControl ) const
436{
437 const auto rect = qskSubControlRect( this, skinnable, subControl );
438 return updateBoxNode( skinnable, node, rect, subControl );
439}
440
441QSGNode* QskSkinlet::updateBoxNode( const QskSkinnable* skinnable,
442 QSGNode* node, const QRectF& rect, QskAspect::Subcontrol subControl )
443{
444 const auto fillGradient = skinnable->gradientHint( subControl );
445 return updateBoxNode( skinnable, node, rect, fillGradient, subControl );
446}
447
448QSGNode* QskSkinlet::updateBoxNode( const QskSkinnable* skinnable,
449 QSGNode* node, const QRectF& rect, const QskGradient& fillGradient,
450 QskAspect::Subcontrol subControl )
451{
452 const auto margins = skinnable->marginHint( subControl );
453
454 const auto boxRect = rect.marginsRemoved( margins );
455 if ( boxRect.isEmpty() )
456 return nullptr;
457
458 const auto borderMetrics = skinnable->boxBorderMetricsHint( subControl );
459 const auto borderColors = skinnable->boxBorderColorsHint( subControl );
460 const auto shape = skinnable->boxShapeHint( subControl );
461 const auto shadowMetrics = skinnable->shadowMetricsHint( subControl );
462 const auto shadowColor = skinnable->shadowColorHint( subControl );
463
464 return qskUpdateBoxNode( skinnable, node,
465 boxRect, shape, borderMetrics, borderColors, fillGradient,
466 shadowMetrics, shadowColor );
467}
468
469QSGNode* QskSkinlet::updateBoxNode(
470 const QskSkinnable* skinnable, QSGNode* node, const QRectF& rect,
471 const QskBoxShapeMetrics& shape, const QskBoxBorderMetrics& borderMetrics,
472 const QskBoxBorderColors& borderColors, const QskGradient& fillGradient )
473{
474 return qskUpdateBoxNode( skinnable, node,
475 rect, shape, borderMetrics, borderColors, fillGradient,
476 QskShadowMetrics(), QColor() );
477}
478
479QSGNode* QskSkinlet::updateBoxNode( const QskSkinnable* skinnable,
480 QSGNode* node, const QRectF& rect, const QskBoxHints& hints )
481{
482 return qskUpdateBoxNode( skinnable, node, rect,
483 hints.shape, hints.borderMetrics, hints.borderColors, hints.gradient,
484 hints.shadowMetrics, hints.shadowColor );
485}
486
487QSGNode* QskSkinlet::updateInterpolatedBoxNode(
488 const QskSkinnable* skinnable, QSGNode* node, const QRectF& rect,
489 QskAspect aspect1, QskAspect aspect2, qreal ratio )
490{
491 QskBoxHints boxHints;
492 QRectF r;
493
494 ratio = qBound( 0.0, ratio, 1.0 );
495
496 if ( qFuzzyIsNull( ratio ) )
497 {
498 const auto margins = skinnable->marginHint( aspect1 );
499 r = rect.marginsRemoved( margins );
500
501 boxHints = skinnable->boxHints( aspect1 ).toAbsolute( r.size() );
502 }
503 else if ( qFuzzyCompare( ratio, 1.0 ) )
504 {
505 const auto margins = skinnable->marginHint( aspect2 );
506 r = rect.marginsRemoved( margins );
507
508 boxHints = skinnable->boxHints( aspect2 ).toAbsolute( r.size() );
509 }
510 else
511 {
512 QskMargins margins = skinnable->marginHint( aspect1 );
513 margins = margins.interpolated( skinnable->marginHint( aspect2 ), ratio );
514
515 r = rect.marginsRemoved( margins );
516
517 const auto boxHints1 = skinnable->boxHints( aspect1 ).toAbsolute( r.size() );
518 const auto boxHints2 = skinnable->boxHints( aspect2 ).toAbsolute( r.size() );
519
520 boxHints = boxHints1.interpolated( boxHints2, ratio );
521 }
522
523 return QskSkinlet::updateBoxNode( skinnable, node, r, boxHints );
524}
525
526QSGNode* QskSkinlet::updateArcNode( const QskSkinnable* skinnable,
527 QSGNode* node, QskAspect::Subcontrol subControl ) const
528{
529 const auto rect = qskSubControlRect( this, skinnable, subControl );
530 return updateArcNode( skinnable, node, rect, subControl );
531}
532
533QSGNode* QskSkinlet::updateArcNode( const QskSkinnable* skinnable,
534 QSGNode* node, const QRectF& rect, QskAspect::Subcontrol subControl )
535{
536 const auto fillGradient = skinnable->gradientHint( subControl );
537 return updateArcNode( skinnable, node, rect, fillGradient, subControl );
538}
539
540QSGNode* QskSkinlet::updateArcNode( const QskSkinnable* skinnable,
541 QSGNode* node, const QRectF& rect, const QskGradient& fillGradient,
542 QskAspect::Subcontrol subControl )
543{
544 const auto metrics = skinnable->arcMetricsHint( subControl );
545 const auto r = rect.marginsRemoved( skinnable->marginHint( subControl ) );
546
547 const qreal borderWidth = skinnable->metric( subControl | QskAspect::Border );
548
549 QColor borderColor;
550 if ( borderWidth > 0.0 )
551 borderColor = skinnable->color( subControl | QskAspect::Border );
552
553 return qskUpdateArcNode( skinnable, node,
554 r, borderWidth, borderColor, fillGradient, metrics );
555}
556
557QSGNode* QskSkinlet::updateArcNode(
558 const QskSkinnable* skinnable, QSGNode* node, const QRectF& rect,
559 qreal borderWidth, const QColor& borderColor,
560 const QskGradient& fillGradient, const QskArcMetrics& metrics )
561{
562 return qskUpdateArcNode( skinnable, node, rect,
563 borderWidth, borderColor, fillGradient, metrics );
564}
565
566QSGNode* QskSkinlet::updateArcNode( const QskSkinnable* skinnable,
567 QSGNode* node, qreal startAngle, qreal spanAngle,
568 QskAspect::Subcontrol subControl ) const
569{
570 const auto rect = qskSubControlRect( this, skinnable, subControl );
571
572 return updateArcNode( skinnable, node,
573 rect, startAngle, spanAngle, subControl );
574}
575
576QSGNode* QskSkinlet::updateArcNode( const QskSkinnable* skinnable,
577 QSGNode* node, const QRectF& rect, qreal startAngle, qreal spanAngle,
578 QskAspect::Subcontrol subControl )
579{
580 const auto fillGradient = skinnable->gradientHint( subControl );
581 return updateArcNode( skinnable, node, rect,
582 fillGradient, startAngle, spanAngle, subControl );
583}
584
585QSGNode* QskSkinlet::updateArcNode( const QskSkinnable* skinnable,
586 QSGNode* node, const QRectF& rect, const QskGradient& fillGradient,
587 qreal startAngle, qreal spanAngle, QskAspect::Subcontrol subControl )
588{
589 auto arcMetrics = skinnable->arcMetricsHint( subControl );
590 arcMetrics.setStartAngle( startAngle );
591 arcMetrics.setSpanAngle( spanAngle );
592
593 const qreal borderWidth = skinnable->metric( subControl | QskAspect::Border );
594
595 QColor borderColor;
596 if ( borderWidth > 0.0 )
597 borderColor = skinnable->color( subControl | QskAspect::Border );
598
599 const auto r = rect.marginsRemoved( skinnable->marginHint( subControl ) );
600 return updateArcNode( skinnable, node, r,
601 borderWidth, borderColor, fillGradient, arcMetrics );
602}
603
604QSGNode* QskSkinlet::updateLineNode( const QskSkinnable* skinnable,
605 QSGNode* node, const QLineF& line, QskAspect::Subcontrol subControl )
606{
607 auto lineStipple = skinnable->stippleMetricsHint( subControl );
608 if ( !lineStipple.isValid() )
609 lineStipple = Qt::SolidLine;
610
611 const auto lineWidth = skinnable->metric( subControl | QskAspect::Size );
612 const auto lineColor = skinnable->color( subControl );
613
614 return qskUpdateLineNode( skinnable, node,
615 lineColor, lineWidth, lineStipple, line );
616}
617
618QSGNode* QskSkinlet::updateLinesNode( const QskSkinnable* skinnable,
619 QSGNode* node, const QVector< QLineF >& lines, QskAspect::Subcontrol subControl )
620{
621 auto lineStipple = skinnable->stippleMetricsHint( subControl );
622 if ( !lineStipple.isValid() )
623 lineStipple = Qt::SolidLine;
624
625 const auto lineWidth = skinnable->metric( subControl | QskAspect::Size );
626 const auto lineColor = skinnable->color( subControl );
627
628 return qskUpdateLinesNode( skinnable, node,
629 lineColor, lineWidth, lineStipple, lines );
630}
631
632QSGNode* QskSkinlet::updateBoxClipNode( const QskSkinnable* skinnable,
633 QSGNode* node, QskAspect::Subcontrol subControl ) const
634{
635 const auto rect = qskSubControlRect( this, skinnable, subControl );
636 return updateBoxClipNode( skinnable, node, rect, subControl );
637}
638
639QSGNode* QskSkinlet::updateBoxClipNode( const QskSkinnable* skinnable,
640 QSGNode* node, const QRectF& rect, QskAspect::Subcontrol subControl )
641{
642 auto clipNode = QskSGNode::ensureNode< QskClipNode >( node );
643
644 const auto margins = skinnable->marginHint( subControl );
645
646 const auto clipRect = rect.marginsRemoved( margins );
647 if ( clipRect.isEmpty() )
648 {
649 clipNode->setRect( clipRect );
650 }
651 else
652 {
653 auto borderMetrics = skinnable->boxBorderMetricsHint( subControl );
654 borderMetrics = borderMetrics.toAbsolute( clipRect.size() );
655
656 auto shape = skinnable->boxShapeHint( subControl );
657 shape = shape.toAbsolute( clipRect.size() );
658
659 const auto window = qskWindowOfSkinnable( skinnable );
660 clipNode->setBox( window, clipRect, shape, borderMetrics );
661 }
662
663 return clipNode;
664}
665
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,
669 const QskTextColors& textColors, Qsk::TextStyle textStyle )
670{
671 return qskUpdateTextNode( skinnable, node, rect, alignment,
672 text, font, textOptions, textColors, textStyle );
673}
674
675QSGNode* QskSkinlet::updateTextNode(
676 const QskSkinnable* skinnable, QSGNode* node,
677 const QRectF& rect, Qt::Alignment alignment,
678 const QString& text, QskAspect::Subcontrol subControl )
679{
680 const auto textOptions = skinnable->textOptionsHint( subControl );
681
682 return updateTextNode( skinnable, node, rect, alignment,
683 textOptions, text, subControl );
684}
685
686QSGNode* QskSkinlet::updateTextNode( const QskSkinnable* skinnable,
687 QSGNode* node, const QRectF& rect,
688 Qt::Alignment alignment, const QskTextOptions& textOptions,
689 const QString& text, QskAspect::Subcontrol subControl )
690{
691 if ( text.isEmpty() || rect.isEmpty() )
692 return nullptr;
693
694 const auto colors = qskTextColors( skinnable, subControl );
695
696 auto style = Qsk::Normal;
697 if ( colors.styleColor().isValid() )
698 {
699 style = skinnable->flagHint< Qsk::TextStyle >(
700 subControl | QskAspect::Style, Qsk::Normal );
701 }
702
703 const auto font = skinnable->effectiveFont( subControl );
704
705 return qskUpdateTextNode( skinnable, node, rect, alignment,
706 text, font, textOptions, colors, style );
707}
708
709QSGNode* QskSkinlet::updateTextNode(
710 const QskSkinnable* skinnable, QSGNode* node,
711 const QString& text, QskAspect::Subcontrol subControl ) const
712{
713 const auto rect = qskSubControlRect( this, skinnable, subControl );
714 const auto alignment = skinnable->alignmentHint( subControl, Qt::AlignLeft );
715
716 return updateTextNode( skinnable, node,
717 rect, alignment, text, subControl );
718}
719
720QSGNode* QskSkinlet::updateSymbolNode(
721 const QskSkinnable* skinnable, QSGNode* node,
722 QskAspect::Subcontrol subControl ) const
723{
724 return updateGraphicNode( skinnable, node,
725 skinnable->symbolHint( subControl ), subControl );
726}
727
728QSGNode* QskSkinlet::updateGraphicNode(
729 const QskSkinnable* skinnable, QSGNode* node,
730 const QskGraphic& graphic, QskAspect::Subcontrol subControl,
731 Qt::Orientations mirrored ) const
732{
733 auto rect = qskSubControlRect( this, skinnable, subControl );
734 rect = rect.marginsRemoved( skinnable->marginHint( subControl ) );
735
736 const auto alignment = skinnable->alignmentHint( subControl, Qt::AlignCenter );
737 const auto colorFilter = skinnable->effectiveGraphicFilter( subControl );
738
739 return updateGraphicNode( skinnable, node,
740 graphic, colorFilter, rect, alignment, mirrored );
741}
742
743QSGNode* QskSkinlet::updateGraphicNode(
744 const QskSkinnable* skinnable, QSGNode* node,
745 const QskGraphic& graphic, const QskColorFilter& colorFilter,
746 const QRectF& rect, Qt::Alignment alignment, Qt::Orientations mirrored )
747{
748 if ( graphic.isNull() )
749 return nullptr;
750
751 const auto size = graphic.defaultSize().scaled(
752 rect.size(), Qt::KeepAspectRatio );
753
754 const auto r = qskAlignedRectF( rect, size, alignment );
755 return qskUpdateGraphicNode( skinnable, node, graphic, colorFilter, r, mirrored );
756}
757
758QSGNode* QskSkinlet::updateGraphicNode(
759 const QskSkinnable* skinnable, QSGNode* node,
760 const QskGraphic& graphic, const QskColorFilter& colorFilter,
761 const QRectF& rect, Qt::Orientations mirrored )
762{
763 if ( graphic.isNull() )
764 return nullptr;
765
766 return qskUpdateGraphicNode( skinnable, node, graphic, colorFilter, rect, mirrored );
767}
768
769int QskSkinlet::sampleIndexAt( const QskSkinnable* skinnable,
770 const QRectF& rect, QskAspect::Subcontrol subControl, const QPointF& pos ) const
771{
772 // slow default implementation to be overloaded when having many cells
773
774 const auto count = sampleCount( skinnable, subControl );
775
776 for ( int i = 0; i < count; i++ )
777 {
778 const auto r = sampleRect( skinnable, rect, subControl, i );
779 if ( r.contains( pos ) )
780 return i;
781 }
782
783 return -1;
784}
785
786QSGNode* QskSkinlet::updateSeriesNode( const QskSkinnable* skinnable,
787 QskAspect::Subcontrol subControl, QSGNode* rootNode ) const
788{
789 auto node = rootNode ? rootNode->firstChild() : nullptr;
790 QSGNode* lastNode = nullptr;
791
792 const auto count = sampleCount( skinnable, subControl );
793
794 for( int i = 0; i < count; i++ )
795 {
796 QSGNode* newNode = nullptr;
797
798 {
799 const auto newStates = sampleStates( skinnable, subControl, i );
800
801 QskSkinStateChanger stateChanger( skinnable );
802 stateChanger.setStates( newStates, i );
803
804 newNode = updateSampleNode( skinnable, subControl, i, node );
805 }
806
807 if ( newNode )
808 {
809 if ( newNode == node )
810 {
811 node = node->nextSibling();
812 }
813 else
814 {
815 if ( rootNode == nullptr )
816 rootNode = new QSGNode();
817
818 if ( node )
819 rootNode->insertChildNodeBefore( newNode, node );
820 else
821 rootNode->appendChildNode( newNode );
822 }
823
824 lastNode = newNode;
825 }
826 }
827
828 QskSGNode::removeAllChildNodesAfter( rootNode, lastNode );
829
830 return rootNode;
831}
832
833QSGNode* QskSkinlet::updateSampleNode( const QskSkinnable*,
834 QskAspect::Subcontrol, int index, QSGNode* ) const
835{
836 Q_UNUSED( index )
837 return nullptr;
838}
839
840QskAspect::States QskSkinlet::sampleStates(
841 const QskSkinnable* skinnable, QskAspect::Subcontrol, int index ) const
842{
843 Q_UNUSED( index )
844 return skinnable->skinStates();
845}
846
847QVariant QskSkinlet::sampleAt( const QskSkinnable*,
848 QskAspect::Subcontrol, int index ) const
849{
850 Q_UNUSED( index )
851 return QVariant();
852}
853
854QSizeF QskSkinlet::hintWithoutConstraint(
855 const QSizeF& hint, const QSizeF& constraint ) const
856{
857 /*
858 This method is useful in situations, where a hint has been calculated
859 from a constraint and we want to return the calculated part only
860 */
861 QSizeF h;
862
863 if ( constraint.width() < 0.0 )
864 h.setWidth( hint.width() );
865
866 if ( constraint.height() < 0.0 )
867 h.setHeight( hint.height() );
868
869 return h;
870}
871
872#include "moc_QskSkinlet.cpp"
Lookup key for a QskSkinHintTable.
Definition QskAspect.h:15
Subcontrol
For use within the rendering or lay-outing of a specific QskSkinnable.
Definition QskAspect.h:104
Base class of all controls.
Definition QskControl.h:23
QskGradient background
Definition QskControl.h:41
A paint device for scalable graphics.
Definition QskGraphic.h:28
QRectF rect
Definition QskItem.h:22
@ DebugForceBackground
Definition QskItem.h:59
@ PreferRasterForTextures
Definition QskItem.h:57
bool isValid() const
Describes the rendering interface of a QskControl. Change the skinlet to change the appearance of the...
Definition QskSkinlet.h:34
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.