QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskSkinnable.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskSkinnable.h"
7
8#include "QskAnimationHint.h"
9#include "QskArcMetrics.h"
10#include "QskAspect.h"
11#include "QskColorFilter.h"
12#include "QskControl.h"
13#include "QskHintAnimator.h"
14#include "QskMargins.h"
15#include "QskSkinManager.h"
16#include "QskSkin.h"
17#include "QskSkinHintTable.h"
18#include "QskSkinTransition.h"
19#include "QskSkinlet.h"
20#include "QskWindow.h"
21
22#include "QskBoxShapeMetrics.h"
23#include "QskBoxBorderMetrics.h"
24#include "QskBoxBorderColors.h"
25#include "QskShadowMetrics.h"
26#include "QskStippleMetrics.h"
27#include "QskBoxHints.h"
28#include "QskGradient.h"
29#include "QskTextOptions.h"
30#include "QskGraphic.h"
31#include "QskFontRole.h"
32
33#include <qfont.h>
34#include <qfontmetrics.h>
35#include <map>
36
37#define DEBUG_MAP 0
38#define DEBUG_ANIMATOR 0
39#define DEBUG_STATE 0
40
41static inline bool qskIsControl( const QskSkinnable* skinnable )
42{
43 return skinnable->metaObject()->inherits( &QskControl::staticMetaObject );
44}
45
46static inline bool qskSetFlag( QskSkinnable* skinnable,
47 const QskAspect aspect, int flag )
48{
49 return skinnable->setSkinHint( aspect, QVariant( flag ) );
50}
51
52static inline int qskFlag( const QskSkinnable* skinnable,
53 const QskAspect aspect, QskSkinHintStatus* status = nullptr )
54{
55 return skinnable->effectiveSkinHint( aspect, status ).toInt();
56}
57
58static inline bool qskSetMetric( QskSkinnable* skinnable,
59 const QskAspect aspect, const QVariant& metric )
60{
61 return skinnable->setSkinHint( aspect | QskAspect::Metric, metric );
62}
63
64static inline bool qskMoveMetric( QskSkinnable* skinnable,
65 const QskAspect aspect, const QVariant& metric )
66{
67 return skinnable->moveSkinHint( aspect | QskAspect::Metric, metric );
68}
69
70template< typename T >
71static inline bool qskSetMetric( QskSkinnable* skinnable,
72 QskAspect aspect, const T& metric )
73{
74 return qskSetMetric( skinnable, aspect, QVariant::fromValue( metric ) );
75}
76
77template< typename T >
78static inline bool qskMoveMetric( QskSkinnable* skinnable,
79 QskAspect aspect, const T& metric )
80{
81 return qskMoveMetric( skinnable, aspect, QVariant::fromValue( metric ) );
82}
83
84template< typename T >
85static inline T qskMetric( const QskSkinnable* skinnable,
86 QskAspect aspect, QskSkinHintStatus* status = nullptr )
87{
88 return skinnable->effectiveSkinHint(
89 aspect | QskAspect::Metric, status ).value< T >();
90}
91
92static inline bool qskSetColor( QskSkinnable* skinnable,
93 const QskAspect aspect, const QVariant& color )
94{
95 return skinnable->setSkinHint( aspect | QskAspect::Color, color );
96}
97
98static inline bool qskMoveColor( QskSkinnable* skinnable,
99 const QskAspect aspect, const QVariant& color )
100{
101 return skinnable->moveSkinHint( aspect | QskAspect::Color, color );
102}
103
104template< typename T >
105static inline bool qskSetColor( QskSkinnable* skinnable,
106 const QskAspect aspect, const T& color )
107{
108 return qskSetColor( skinnable, aspect, QVariant::fromValue( color ) );
109}
110
111template< typename T >
112static inline bool qskMoveColor( QskSkinnable* skinnable,
113 const QskAspect aspect, const T& color )
114{
115 return qskMoveColor( skinnable, aspect, QVariant::fromValue( color ) );
118template< typename T >
119static inline T qskColor( const QskSkinnable* skinnable,
120 QskAspect aspect, QskSkinHintStatus* status = nullptr )
121{
122 return skinnable->effectiveSkinHint(
123 aspect | QskAspect::Color, status ).value< T >();
124}
125
126static inline constexpr QskAspect qskAnimatorAspect( const QskAspect aspect )
127{
128 /*
129 We do not need the extra bits that would slow down resolving
130 the effective aspect in animatedHint.
131 */
132
133 return aspect.type() | aspect.subControl() | aspect.primitive();
134}
135
136static inline void qskTriggerUpdates( QskAspect aspect, QQuickItem* item )
137{
138 /*
139 To put the hint into effect we have to call the usual suspects:
140
141 - resetImplicitSize
142 - polish
143 - update
144
145 The following code decides about these calls based on type/primitive
146 of the aspect. It can be expected, that it results in more calls
147 than what would be mandatory and in rare cases we might even miss necessary
148 calls. This has to be fixed by doing the call manually in the specific
149 controls.
150 */
151
152 if ( item == nullptr || aspect.isAnimator() )
153 return;
154
155 item->update(); // always
156
157 auto control = qskControlCast( item );
158 if ( control == nullptr )
159 return;
160
161 bool maybeLayout = false;
162
163 switch( aspect.type() )
164 {
165 using A = QskAspect;
166
167 case A::Metric:
168 {
169 if ( aspect.metricPrimitive() != A::Position )
170 {
171 control->resetImplicitSize();
172 maybeLayout = true;
173 }
174
175 break;
176 }
177
178 case A::Color:
179 {
180 break;
181 }
182
183 case A::NoType:
184 {
185 switch( aspect.primitive() )
186 {
187 case A::GraphicRole:
188 case A::FontRole:
189 {
190 break;
191 }
192 case A::Alignment:
193 {
194 maybeLayout = true;
195 break;
196 }
197 default:
198 {
199 control->resetImplicitSize();
200 maybeLayout = true;
201 }
202 }
203 }
204 }
205
206 if ( maybeLayout && control->hasChildItems() )
207 {
208 if ( control->polishOnResize() || control->autoLayoutChildren() )
209 control->polish();
210 }
211}
212
213static inline QskAspect qskSubstitutedAspect(
214 const QskSkinnable* skinnable, QskAspect aspect )
215{
216#if 0
217 if ( aspect.hasStates() )
218 {
219 qWarning() << "QskSkinnable::(re)setSkinHint: setting hints with states "
220 "is discouraged - use QskSkinTableEditor if you are sure, that you need this.";
221
222 qWarning() << "QskAspect:" << aspect.stateless()
223 << skinnable->skinStatesAsPrintable( aspect.states() );
224
225#if 0
226 aspect.clearStates();
227#endif
228 }
229#endif
230
231 aspect.setSubcontrol( skinnable->effectiveSubcontrol( aspect.subControl() ) );
232 return aspect;
233}
234
235class QskSkinnable::PrivateData
236{
237 public:
238 ~PrivateData()
239 {
240 if ( hasLocalSkinlet )
241 {
242 if ( skinlet && skinlet->isOwnedBySkinnable() )
243 delete skinlet;
244 }
245
246 delete subcontrolProxies;
247 }
248
249 QskSkinHintTable hintTable;
250 QskHintAnimatorTable animators;
251
252 int sampleIndex = -1; // for the ugly QskSkinStateChanger hack
253
254 typedef std::map< QskAspect::Subcontrol, QskAspect::Subcontrol > ProxyMap;
255 ProxyMap* subcontrolProxies = nullptr;
256
257 const QskSkinlet* skinlet = nullptr;
258
259 QskAspect::States skinStates;
260 bool hasLocalSkinlet = false;
261};
262
264 : m_data( new PrivateData() )
265{
266}
267
271
273{
274 if ( skinlet == m_data->skinlet )
275 {
276 if ( skinlet )
277 {
278 // now we don't depend on global skin changes anymore
279 m_data->hasLocalSkinlet = true;
280 }
281 return;
282 }
283
284 if ( m_data->skinlet && m_data->skinlet->isOwnedBySkinnable() )
285 delete m_data->skinlet;
286
287 m_data->skinlet = skinlet;
288 m_data->hasLocalSkinlet = ( skinlet != nullptr );
289
290 if ( auto item = owningItem() )
291 {
292 if ( auto control = qskControlCast( item ) )
293 control->resetImplicitSize();
294
295 item->polish();
296
297 if ( item->flags() & QQuickItem::ItemHasContents )
298 item->update();
299 }
300}
301
303{
304 return m_data->hasLocalSkinlet ? m_data->skinlet : nullptr;
305}
306
308{
309 if ( m_data->skinlet == nullptr )
310 {
311 m_data->skinlet = effectiveSkin()->skinlet( metaObject() );
312 m_data->hasLocalSkinlet = false;
313 }
314
315 return m_data->skinlet;
316}
317
318void QskSkinnable::setSubcontrolProxy(
320{
321 if ( subControl == QskAspect::NoSubcontrol )
322 return; // nonsense, we ignore this
323
324 if ( proxy == QskAspect::NoSubcontrol || subControl == proxy )
325 {
326 resetSubcontrolProxy( subControl );
327 return;
328 }
329
330 if ( m_data->subcontrolProxies == nullptr )
331 m_data->subcontrolProxies = new PrivateData::ProxyMap();
332
333 ( *m_data->subcontrolProxies )[ subControl ] = proxy;
334}
335
336void QskSkinnable::resetSubcontrolProxy( QskAspect::Subcontrol subcontrol )
337{
338 if ( auto& proxies = m_data->subcontrolProxies )
339 {
340 auto it = proxies->find( subcontrol );
341 if ( it != proxies->end() )
342 {
343 proxies->erase( it );
344 if ( proxies->empty() )
345 {
346 delete proxies;
347 proxies = nullptr;
348 }
349 }
350 }
351}
352
353QskAspect::Subcontrol QskSkinnable::subcontrolProxy( QskAspect::Subcontrol subControl ) const
354{
355 if ( const auto proxies = m_data->subcontrolProxies )
356 {
357 auto it = proxies->find( subControl );
358 if ( it != proxies->end() )
359 return it->second;
360 }
361
362 return QskAspect::NoSubcontrol;
363}
364
366{
367 return m_data->hintTable;
368}
369
371{
372 return m_data->hintTable;
373}
374
375bool QskSkinnable::setFlagHint( const QskAspect aspect, int flag )
376{
377 return qskSetFlag( this, aspect, flag );
378}
379
380bool QskSkinnable::setAlignmentHint( const QskAspect aspect, Qt::Alignment alignment )
381{
382 return qskSetFlag( this, aspect | QskAspect::Alignment, alignment );
383}
384
386{
387 return resetSkinHint( aspect | QskAspect::Alignment );
388}
389
390bool QskSkinnable::setColor( const QskAspect aspect, const QColor& color )
391{
392 return qskSetColor( this, aspect, color );
393}
394
395bool QskSkinnable::setColor( const QskAspect aspect, Qt::GlobalColor color )
396{
397 return qskSetColor( this, aspect, QColor( color ) );
398}
399
400bool QskSkinnable::setColor( const QskAspect aspect, QRgb rgb )
401{
402 return qskSetColor( this, aspect, QColor::fromRgba( rgb ) );
403}
404
405bool QskSkinnable::moveColor( const QskAspect aspect, const QColor& color )
406{
407 return qskMoveColor( this, aspect, color );
408}
409
410bool QskSkinnable::moveColor( const QskAspect aspect, Qt::GlobalColor color )
411{
412 return qskMoveColor( this, aspect, QColor( color ) );
413}
414
415bool QskSkinnable::moveColor( const QskAspect aspect, QRgb rgb )
416{
417 return qskMoveColor( this, aspect, QColor::fromRgba( rgb ) );
418}
419
420QColor QskSkinnable::color( const QskAspect aspect, QskSkinHintStatus* status ) const
421{
422 return qskColor< QColor >( this, aspect, status );
423}
424
425bool QskSkinnable::setMetric( const QskAspect aspect, qreal metric )
426{
427 return qskSetMetric( this, aspect, metric );
428}
429
430bool QskSkinnable::moveMetric( QskAspect aspect, qreal metric )
431{
432 return qskMoveMetric( this, aspect, metric );
433}
434
435qreal QskSkinnable::metric( const QskAspect aspect, QskSkinHintStatus* status ) const
436{
437 return qskMetric< qreal >( this, aspect, status );
438}
439
440qreal QskSkinnable::metric( QskAspect aspect, qreal defaultValue ) const
441{
442 QskSkinHintStatus status;
443
444 const auto value = qskMetric< qreal >( this, aspect, &status );
445 return status.isValid() ? value : defaultValue;
446}
447
448bool QskSkinnable::setPositionHint( QskAspect aspect, qreal position )
449{
450 return qskSetMetric( this, aspect | QskAspect::Position, position );
451}
452
453bool QskSkinnable::movePositionHint( QskAspect aspect, qreal position )
454{
455 return qskMoveMetric( this, aspect | QskAspect::Position, position );
456}
457
458bool QskSkinnable::movePositionHint( QskAspect aspect, qreal from, qreal to )
459{
460 return moveSkinHint( aspect | QskAspect::Metric | QskAspect::Position,
461 QVariant::fromValue( from ), QVariant::fromValue( to ) );
462}
463
464bool QskSkinnable::resetPositionHint( QskAspect aspect )
465{
466 return resetMetric( aspect | QskAspect::Position );
467}
468
469qreal QskSkinnable::positionHint( QskAspect aspect, QskSkinHintStatus* status ) const
470{
471 return qskMetric< qreal >( this, aspect | QskAspect::Position, status );
472}
473
475 const QskAspect aspect, qreal width, qreal height )
476{
477 return qskSetMetric( this, aspect | QskAspect::StrutSize, QSizeF( width, height ) );
478}
479
480bool QskSkinnable::setStrutSizeHint( const QskAspect aspect, const QSizeF& size )
481{
482 return qskSetMetric( this, aspect | QskAspect::StrutSize, size );
483}
484
486{
487 return resetMetric( aspect | QskAspect::StrutSize );
488}
489
491 const QskAspect aspect, QskSkinHintStatus* status ) const
492{
493 return qskMetric< QSizeF >( this, aspect | QskAspect::StrutSize, status );
494}
495
496bool QskSkinnable::setMarginHint( const QskAspect aspect, qreal margins )
497{
498 return qskSetMetric( this, aspect | QskAspect::Margin, QskMargins( margins ) );
499}
500
501bool QskSkinnable::setMarginHint( const QskAspect aspect, const QMarginsF& margins )
502{
503 return qskSetMetric( this, aspect | QskAspect::Margin, QskMargins( margins ) );
504}
505
507{
508 return resetMetric( aspect | QskAspect::Margin );
509}
510
512 const QskAspect aspect, QskSkinHintStatus* status ) const
513{
514 return qskMetric< QskMargins >( this, aspect | QskAspect::Margin, status );
515}
516
517bool QskSkinnable::setPaddingHint( const QskAspect aspect, qreal padding )
518{
519 return qskSetMetric( this, aspect | QskAspect::Padding, QskMargins( padding ) );
520}
521
522bool QskSkinnable::setPaddingHint( const QskAspect aspect, const QMarginsF& padding )
523{
524 return qskSetMetric( this, aspect | QskAspect::Padding, QskMargins( padding ) );
525}
526
528{
529 return resetMetric( aspect | QskAspect::Padding );
530}
531
533 const QskAspect aspect, QskSkinHintStatus* status ) const
534{
535 return qskMetric< QskMargins >( this, aspect | QskAspect::Padding, status );
536}
537
539 const QskAspect aspect, const QskGradient& gradient )
540{
541 return qskSetColor( this, aspect, gradient );
542}
543
545 const QskAspect aspect, QskSkinHintStatus* status ) const
546{
547 return qskColor< QskGradient >( this, aspect, status );
548}
549
551 const QskAspect aspect, const QskBoxShapeMetrics& shape )
552{
553 return qskSetMetric( this, aspect | QskAspect::Shape, shape );
554}
555
557{
558 return resetMetric( aspect | QskAspect::Shape );
559}
560
562 const QskAspect aspect, QskSkinHintStatus* status ) const
563{
564 return qskMetric< QskBoxShapeMetrics >(
565 this, aspect | QskAspect::Shape, status );
566}
567
569 const QskAspect aspect, const QskBoxBorderMetrics& border )
570{
571 return qskSetMetric( this, aspect | QskAspect::Border, border );
572}
573
575{
576 return resetMetric( aspect | QskAspect::Border );
577}
578
580 const QskAspect aspect, QskSkinHintStatus* status ) const
581{
582 return qskMetric< QskBoxBorderMetrics >(
583 this, aspect | QskAspect::Border, status );
584}
585
587 const QskAspect aspect, const QskBoxBorderColors& colors )
588{
589 return qskSetColor( this, aspect | QskAspect::Border, colors );
590}
591
593{
594 return resetColor( aspect | QskAspect::Border );
595}
596
598 const QskAspect aspect, QskSkinHintStatus* status ) const
599{
600 return qskColor< QskBoxBorderColors >(
601 this, aspect | QskAspect::Border, status );
602}
603
604bool QskSkinnable::setShadowMetricsHint(
605 QskAspect aspect, const QskShadowMetrics& metrics )
606{
607 return qskSetMetric( this, aspect | QskAspect::Shadow, metrics );
608}
609
610bool QskSkinnable::resetShadowMetricsHint( QskAspect aspect )
611{
612 return resetMetric( aspect | QskAspect::Shadow );
613}
614
615QskShadowMetrics QskSkinnable::shadowMetricsHint(
616 QskAspect aspect, QskSkinHintStatus* status ) const
617{
618 return qskMetric< QskShadowMetrics >(
619 this, aspect | QskAspect::Shadow, status );
620}
621
622bool QskSkinnable::setShadowColorHint( QskAspect aspect, const QColor& color )
623{
624 return qskSetColor( this, aspect | QskAspect::Shadow, color );
625}
626
627bool QskSkinnable::resetShadowColorHint( QskAspect aspect )
628{
629 return resetColor( aspect | QskAspect::Shadow );
630}
631
632QColor QskSkinnable::shadowColorHint( QskAspect aspect, QskSkinHintStatus* status ) const
633{
634 return qskColor< QColor >( this, aspect | QskAspect::Shadow, status );
635}
636
637QskBoxHints QskSkinnable::boxHints( QskAspect aspect ) const
638{
639 return QskBoxHints(
640 boxShapeHint( aspect ), boxBorderMetricsHint( aspect ),
641 boxBorderColorsHint( aspect ), gradientHint( aspect ),
642 shadowMetricsHint( aspect ), shadowColorHint( aspect ) );
643}
644
645bool QskSkinnable::setArcMetricsHint(
646 const QskAspect aspect, const QskArcMetrics& arc )
647{
648 return qskSetMetric( this, aspect | QskAspect::Shape, arc );
649}
650
651bool QskSkinnable::resetArcMetricsHint( const QskAspect aspect )
652{
653 return resetMetric( aspect | QskAspect::Shape );
654}
655
656QskArcMetrics QskSkinnable::arcMetricsHint(
657 const QskAspect aspect, QskSkinHintStatus* status ) const
658{
659 return qskMetric< QskArcMetrics >(
660 this, aspect | QskAspect::Shape, status );
661}
662
663bool QskSkinnable::setStippleMetricsHint(
664 QskAspect aspect, const QskStippleMetrics& metrics )
665{
666 return qskSetMetric( this, aspect | QskAspect::Style, metrics );
667}
668
669bool QskSkinnable::resetStippleMetricsHint( QskAspect aspect )
670{
671 return resetMetric( aspect | QskAspect::Style );
672}
673
674QskStippleMetrics QskSkinnable::stippleMetricsHint(
675 QskAspect aspect, QskSkinHintStatus* status ) const
676{
677 return qskMetric< QskStippleMetrics >(
678 this, aspect | QskAspect::Style, status );
679}
680
681bool QskSkinnable::setSpacingHint( const QskAspect aspect, qreal spacing )
682{
683 return qskSetMetric( this, aspect | QskAspect::Spacing, spacing );
684}
685
687{
688 return resetMetric( aspect | QskAspect::Spacing );
689}
690
692 const QskAspect aspect, QskSkinHintStatus* status ) const
693{
694 return qskMetric< qreal >( this, aspect | QskAspect::Spacing, status );
695}
696
697bool QskSkinnable::setTextOptionsHint(
698 const QskAspect aspect, const QskTextOptions& options )
699{
700 return setSkinHint( aspect | QskAspect::Option,
701 QVariant::fromValue( options ) );
702}
703
704bool QskSkinnable::resetTextOptionsHint( const QskAspect aspect )
705{
706 return resetSkinHint( aspect | QskAspect::Option );
707}
708
709QskTextOptions QskSkinnable::textOptionsHint(
710 const QskAspect aspect, QskSkinHintStatus* status ) const
711{
712 return effectiveSkinHint(
713 aspect | QskAspect::Option, status ).value< QskTextOptions >();
714}
715
717 const QskAspect aspect, const QskFontRole& role )
718{
719 return setSkinHint( aspect | QskAspect::FontRole,
720 QVariant::fromValue( role ) );
721}
722
724{
725 return resetSkinHint( aspect | QskAspect::FontRole );
726}
727
729 const QskAspect aspect, QskSkinHintStatus* status ) const
730{
731 return effectiveSkinHint(
732 aspect | QskAspect::FontRole, status ).value< QskFontRole >();
733}
734
736{
737 const auto hint = effectiveSkinHint( aspect | QskAspect::FontRole );
738 if ( hint.canConvert< QFont >() )
739 {
740 /*
741 The provided skins/controls use font roles only - however
742 application code might want to assign fonts without defining
743 font roles.
744 */
745 return hint.value< QFont >();
746 }
747
748 const auto fontRole = hint.value< QskFontRole >();
749
750 auto font = effectiveSkin()->font( fontRole );
751
752 if ( auto item = owningItem() )
753 {
754 const auto v = QskSkinTransition::animatedFontSize(
755 item->window(), fontRole );
756
757 if ( v.canConvert< int >() )
758 {
759 font.setPixelSize( v.value< int >() );
760 item->update(); // design flaw: see effectiveGraphicFilter
761 }
762 }
763
764 return font;
765}
766
767qreal QskSkinnable::effectiveFontHeight( const QskAspect aspect ) const
768{
769 const QFontMetricsF fm( effectiveFont( aspect ) );
770 return fm.height();
771}
772
773bool QskSkinnable::setGraphicRoleHint( const QskAspect aspect, int role )
774{
775 return qskSetFlag( this, aspect | QskAspect::GraphicRole, role );
776}
777
779{
780 return resetSkinHint( aspect | QskAspect::GraphicRole );
781}
782
784 const QskAspect aspect, QskSkinHintStatus* status ) const
785{
786 return qskFlag( this, aspect | QskAspect::GraphicRole, status );
787}
788
789bool QskSkinnable::setSymbolHint(
790 const QskAspect aspect, const QskGraphic& symbol )
791{
792 return setSkinHint( aspect | QskAspect::Symbol,
793 QVariant::fromValue( symbol ) );
794}
795
796bool QskSkinnable::resetSymbolHint( const QskAspect aspect )
797{
798 return resetSkinHint( aspect | QskAspect::Symbol );
799}
800
801QskGraphic QskSkinnable::symbolHint(
802 const QskAspect aspect, QskSkinHintStatus* status ) const
803{
804 return effectiveSkinHint(
805 aspect | QskAspect::Symbol, status ).value< QskGraphic >();
806}
807
808
810 const QskAspect::Subcontrol subControl ) const
811{
812 /*
813 Usually we find the graphic role and return the related filter
814 from the skin. But as we can't interpolate between graphic roles
815 the corresponding animators interpolate the filters.
816 */
817
818 QskAspect aspect( effectiveSubcontrol( subControl ) | QskAspect::GraphicRole );
819 aspect.setSection( section() );
821
822 QskSkinHintStatus status;
823
824 const auto hint = storedHint( aspect | skinStates(), &status );
825 if ( !status.isValid() )
826 return QskColorFilter();
827
828 aspect.setSubcontrol( status.aspect.subControl() );
829 aspect.setSection( QskAspect::Body );
831
832 {
833 const auto v = animatedHint( aspect, nullptr );
834
835 if ( v.canConvert< QskColorFilter >() )
836 return v.value< QskColorFilter >();
837 }
838
839 if ( auto item = owningItem() )
840 {
841 const auto graphicRole = hint.toInt();
842
843 const auto v = QskSkinTransition::animatedGraphicFilter(
844 item->window(), graphicRole );
845
846 if ( v.canConvert< QskColorFilter >() )
847 {
848#if 1
849 /*
850 Design flaw: the animators for the skin transition do not
851 know about the controls, that are affected from the color
852 filter. As a workaround we schedule the update in the
853 getter: TODO ...
854 */
855 item->update();
856#endif
857 return v.value< QskColorFilter >();
858 }
859 }
860
861 return effectiveSkin()->graphicFilter( hint.toInt() );
862}
863
865 QskAspect aspect, QskAnimationHint hint )
866{
867 aspect.setSubcontrol( effectiveSubcontrol( aspect.subControl() ) );
868 return m_data->hintTable.setAnimation( aspect, hint );
869}
870
872 QskAspect aspect, QskSkinHintStatus* status ) const
873{
874 aspect.setAnimator( true );
875 return effectiveSkinHint( aspect, status ).value< QskAnimationHint >();
876}
877
878bool QskSkinnable::hasAnimationHint( QskAspect aspect ) const
879{
880 return animationHint( aspect ).isValid();
881}
882
884 QskAspect::Type type, QskAspect::Subcontrol subControl,
885 QskAspect::States states, QskSkinHintStatus* status ) const
886{
887#if 0
888 // TODO ...
889 subControl = effectiveSubcontrol( aspect.subControl() );
890#endif
891
892 auto aspect = subControl | type | states;
893 aspect.setAnimator( true );
894
895 QskAnimationHint hint;
896
897 {
898 const auto a = m_data->hintTable.resolvedAnimator( aspect, hint );
899 if ( a.isAnimator() )
900 {
901 if ( status )
902 {
904 status->aspect = a;
905 }
906
907 return hint;
908 }
909 }
910
911 if ( auto skin = effectiveSkin() )
912 {
913 const auto a = skin->hintTable().resolvedAnimator( aspect, hint );
914 if ( a.isAnimator() )
915 {
916 if ( status )
917 {
919 status->aspect = a;
920 }
921
922 return hint;
923 }
924 }
925
926 if ( status )
927 {
929 status->aspect = QskAspect();
930 }
931
932 return hint;
933}
934
935bool QskSkinnable::setSkinHint( QskAspect aspect, const QVariant& hint )
936{
937 aspect = qskSubstitutedAspect( this, aspect );
938
939 if ( m_data->hintTable.setHint( aspect, hint ) )
940 {
941 qskTriggerUpdates( aspect, owningItem() );
942 return true;
943 }
944
945 return false;
946}
947
949{
950 aspect = qskSubstitutedAspect( this, aspect );
951
952 if ( m_data->hintTable.removeHint( aspect ) )
953 {
954 qskTriggerUpdates( aspect, owningItem() );
955 return true;
956 }
957
958 return false;
959}
960
962 QskAspect aspect, QskSkinHintStatus* status ) const
963{
964 aspect.setSubcontrol( effectiveSubcontrol( aspect.subControl() ) );
965
966 if ( !( aspect.isAnimator() || aspect.hasStates() ) )
967 {
968 const auto v = animatedHint( aspect, status );
969 if ( v.isValid() )
970 return v;
971 }
972
973 if ( aspect.section() == QskAspect::Body )
974 aspect.setSection( section() );
975
976 if ( aspect.variation() == QskAspect::NoVariation )
978
979 if ( !aspect.hasStates() )
980 aspect.setStates( skinStates() );
981
982 if ( !aspect.isAnimator() && QskSkinTransition::isRunning() )
983 {
984 /*
985 The skin has changed and the hints are interpolated
986 between the old and the new one over time
987 */
988 const auto v = interpolatedHint( aspect, status );
989 if ( v.isValid() )
990 return v;
991 }
992
993 return storedHint( aspect, status );
994}
995
997{
998 QskSkinHintStatus status;
999
1000 ( void ) effectiveSkinHint( aspect, &status );
1001 return status;
1002}
1003
1004bool QskSkinnable::moveSkinHint( QskAspect aspect,
1005 const QVariant& oldValue, const QVariant& newValue )
1006{
1007 if ( aspect.isAnimator() )
1008 return false;
1009
1010 const bool ok = setSkinHint( aspect, newValue );
1011
1012 if ( ok && oldValue.isValid() && newValue.isValid() )
1013 {
1014 const auto animation = animationHint( aspect );
1015 if ( animation.isValid() )
1016 {
1017 if ( newValue != oldValue )
1018 startTransition( aspect, animation, oldValue, newValue );
1019 }
1020 }
1021
1022 return ok;
1023}
1024
1025bool QskSkinnable::moveSkinHint( QskAspect aspect, const QVariant& value )
1026{
1027 return moveSkinHint( aspect, effectiveSkinHint( aspect ), value );
1028}
1029
1030const QskHintAnimator* QskSkinnable::runningHintAnimator(
1031 QskAspect aspect, int index ) const
1032{
1033 const auto& animators = m_data->animators;
1034
1035 if ( animators.isEmpty() )
1036 return nullptr;
1037
1038 aspect = qskAnimatorAspect( aspect );
1039
1040 auto animator = animators.animator( aspect, index );
1041 if ( animator == nullptr && index >= 0 )
1042 animator = animators.animator( aspect, -1 );
1043
1044 return animator;
1045}
1046
1047QVariant QskSkinnable::animatedHint(
1048 QskAspect aspect, QskSkinHintStatus* status ) const
1049{
1050 QVariant v;
1051
1052 if ( !m_data->animators.isEmpty() )
1053 {
1054 const auto a = qskAnimatorAspect( aspect );
1055
1056 v = m_data->animators.currentValue( a, m_data->sampleIndex );
1057 if ( !v.isValid() && m_data->sampleIndex >= 0 )
1058 v = m_data->animators.currentValue( a, -1 );
1059 }
1060
1061 if ( status && v.isValid() )
1062 {
1064 status->aspect = aspect;
1065 }
1066
1067 return v;
1068}
1069
1070QVariant QskSkinnable::interpolatedHint(
1071 QskAspect aspect, QskSkinHintStatus* status ) const
1072{
1073 if ( !QskSkinTransition::isRunning() || m_data->hintTable.hasHint( aspect ) )
1074 return QVariant();
1075
1076 const auto item = owningItem();
1077 if ( item == nullptr )
1078 return QVariant();
1079
1080 QVariant v;
1081
1082 auto a = aspect;
1083
1084 Q_FOREVER
1085 {
1086 v = QskSkinTransition::animatedHint( item->window(), aspect );
1087
1088 if ( !v.isValid() )
1089 {
1090 if ( const auto topState = aspect.topState() )
1091 {
1092 aspect.clearState( aspect.topState() );
1093 continue;
1094 }
1095
1096 if ( aspect.variation() )
1097 {
1098 // clear the variation bits and restart
1099 aspect = a;
1101
1102 continue;
1103 }
1104 }
1105
1106 if ( a.section() != QskAspect::Body )
1107 {
1108 // try to resolve with the default section
1109
1110 a.setSection( QskAspect::Body );
1111 aspect = a;
1112
1113 continue;
1114 }
1115
1116 break;
1117 }
1118
1119 if ( status && v.isValid() )
1120 {
1122 status->aspect = aspect;
1123 }
1124
1125 return v;
1126}
1127
1128const QVariant& QskSkinnable::storedHint(
1129 QskAspect aspect, QskSkinHintStatus* status ) const
1130{
1131 const auto skin = effectiveSkin();
1132
1133 QskAspect resolvedAspect;
1134
1135 const auto& localTable = m_data->hintTable;
1136 if ( localTable.hasHints() )
1137 {
1138 if ( const auto value = localTable.resolvedHint( aspect, &resolvedAspect ) )
1139 {
1140 if ( status )
1141 {
1143 status->aspect = resolvedAspect;
1144 }
1145 return *value;
1146 }
1147 }
1148
1149 // next we try the hints from the skin
1150
1151 const auto& skinTable = skin->hintTable();
1152 if ( skinTable.hasHints() )
1153 {
1154 if ( const auto value = skinTable.resolvedHint( aspect, &resolvedAspect ) )
1155 {
1156 if ( status )
1157 {
1159 status->aspect = resolvedAspect;
1160 }
1161
1162 return *value;
1163 }
1164
1165 if ( aspect.hasSubcontrol() )
1166 {
1167 // trying to resolve something from the skin default settings
1168
1169 aspect.clearSubcontrol();
1170 aspect.clearStates();
1171
1172 if ( const auto value = skinTable.resolvedHint( aspect, &resolvedAspect ) )
1173 {
1174 if ( status )
1175 {
1177 status->aspect = resolvedAspect;
1178 }
1179
1180 return *value;
1181 }
1182 }
1183 }
1184
1185 if ( status )
1186 {
1188 status->aspect = QskAspect();
1189 }
1190
1191 static QVariant hintInvalid;
1192 return hintInvalid;
1193}
1194
1195bool QskSkinnable::hasSkinState( QskAspect::State state ) const
1196{
1197 return ( m_data->skinStates & state ) == state;
1198}
1199
1200QskAspect::States QskSkinnable::skinStates() const
1201{
1202 return m_data->skinStates;
1203}
1204
1205const char* QskSkinnable::skinStatesAsPrintable() const
1206{
1207 return skinStatesAsPrintable( skinStates() );
1208}
1209
1210const char* QskSkinnable::skinStatesAsPrintable( QskAspect::States states ) const
1211{
1212 QString tmp;
1213
1214 QDebug debug( &tmp );
1215 qskDebugStates( debug, metaObject(), states );
1216
1217 // we should find a better way
1218 static QByteArray bytes[ 10 ];
1219 static int counter = 0;
1220
1221 counter = ( counter + 1 ) % 10;
1222
1223 bytes[ counter ] = tmp.toUtf8();
1224 return bytes[ counter ].constData();
1225}
1226
1227static inline QskMargins qskEffectivePadding( const QskSkinnable* skinnable,
1228 QskAspect aspect, const QSizeF& size, bool inner )
1229{
1230 const auto shape = skinnable->boxShapeHint( aspect ).toAbsolute( size );
1231 const auto borderMetrics = skinnable->boxBorderMetricsHint( aspect );
1232
1233 const qreal left = qMax( shape.radius( Qt::TopLeftCorner ).width(),
1234 shape.radius( Qt::BottomLeftCorner ).width() );
1235
1236 const qreal top = qMax( shape.radius( Qt::TopLeftCorner ).height(),
1237 shape.radius( Qt::TopRightCorner ).height() );
1238
1239 const qreal right = qMax( shape.radius( Qt::TopRightCorner ).width(),
1240 shape.radius( Qt::BottomRightCorner ).width() );
1241
1242 const qreal bottom = qMax( shape.radius( Qt::BottomLeftCorner ).height(),
1243 shape.radius( Qt::BottomRightCorner ).height() );
1244
1245 QskMargins padding( left, top, right, bottom );
1246
1247 // half of the border goes to the inner side
1248 const auto borderMargins = borderMetrics.toAbsolute( size ).widths() * 0.5;
1249
1250 if ( inner )
1251 {
1252 padding -= borderMargins;
1253 }
1254 else
1255 {
1256 // not correct, but to get things started. TODO ...
1257 padding += borderMargins;
1258 }
1259
1260 // sin 45° ceiled : 0.70710678;
1261 padding *= 1.0 - 0.70710678;
1262
1263 return padding.expandedTo( skinnable->paddingHint( aspect ) );
1264}
1265
1267 QskAspect aspect, const QSizeF& outerBoxSize ) const
1268{
1269 return qskEffectivePadding( this, aspect, outerBoxSize, true );
1270}
1271
1273 QskAspect aspect, const QSizeF& outerBoxSize ) const
1274{
1275 const auto pd = qskEffectivePadding( this, aspect, outerBoxSize, true );
1276
1277 return QSizeF( outerBoxSize.width() - pd.width(),
1278 outerBoxSize.height() - pd.height() );
1279}
1280
1282 QskAspect aspect, const QRectF& outerBox ) const
1283{
1284 const auto pd = qskEffectivePadding( this, aspect, outerBox.size(), true );
1285 return outerBox.marginsRemoved( pd );
1286}
1287
1289 QskAspect aspect, const QSizeF& innerBoxSize ) const
1290{
1291 const auto pd = qskEffectivePadding( this, aspect, innerBoxSize, false );
1292 return innerBoxSize.grownBy( pd );
1293}
1294
1296 QskAspect aspect, const QRectF& innerBox ) const
1297{
1298 const auto m = qskEffectivePadding( this, aspect, innerBox.size(), false );
1299 return innerBox.marginsAdded( m );
1300}
1301
1303 const QRectF& contentsRect, QskAspect::Subcontrol subControl ) const
1304{
1305 return effectiveSkinlet()->subControlRect( this, contentsRect, subControl );
1306}
1307
1309 const QRectF& contentsRect, QskAspect::Subcontrol subControl ) const
1310{
1311 return innerBox( subControl, subControlRect( contentsRect, subControl ) );
1312}
1313
1315{
1316 Q_UNUSED( aspect )
1317
1318 if ( auto control = qskControlCast( owningItem() ) )
1319 {
1320 /*
1321 Usually we only need smooth transitions, when state changes
1322 happen while the skinnable is visible. There are few exceptions
1323 like QskPopup::Closed, that is used to slide/fade in.
1324 */
1325
1326 if ( control->flags() & QQuickItem::ItemHasContents )
1327 return control->isInitiallyPainted();
1328
1329 return true;
1330 }
1331
1332 return false;
1333}
1334
1335void QskSkinnable::startTransition( QskAspect aspect,
1336 QskAnimationHint animationHint, const QVariant& from, const QVariant& to )
1337{
1338 startTransition( aspect, -1, animationHint, from, to );
1339}
1340
1341void QskSkinnable::startTransition( QskAspect aspect, int index,
1342 QskAnimationHint animationHint, const QVariant& from, const QVariant& to )
1343{
1344 aspect.setSubcontrol( effectiveSubcontrol( aspect.subControl() ) );
1345 startHintTransition( aspect, index, animationHint, from, to );
1346}
1347
1348void QskSkinnable::startHintTransition( QskAspect aspect, int index,
1349 QskAnimationHint animationHint, const QVariant& from, const QVariant& to )
1350{
1351 if ( animationHint.duration <= 0 || ( from == to ) )
1352 return;
1353
1354 auto control = qskControlCast( owningItem() );
1355 if ( control->window() == nullptr || !isTransitionAccepted( aspect ) )
1356 return;
1357
1358 if ( !QskVariantAnimator::maybeInterpolate( from, to, false ) )
1359 return;
1360
1361 auto v1 = from;
1362 auto v2 = to;
1363
1364 if ( aspect.type() == QskAspect::NoType &&
1365 aspect.primitive() == QskAspect::GraphicRole )
1366 {
1367 const auto skin = effectiveSkin();
1368
1369 v1.setValue( skin->graphicFilter( v1.toInt() ) );
1370 v2.setValue( skin->graphicFilter( v2.toInt() ) );
1371 }
1372
1373 aspect = qskAnimatorAspect( aspect );
1374
1375#if DEBUG_ANIMATOR
1376 qDebug() << aspect << index << animationHint.duration;
1377#endif
1378
1379 auto animator = m_data->animators.animator( aspect, index );
1380 if ( animator && animator->isRunning() )
1381 v1 = animator->currentValue();
1382
1383 m_data->animators.start( control, aspect, index, animationHint, v1, v2 );
1384}
1385
1387{
1388 const auto newState = on
1389 ? ( m_data->skinStates | stateFlag )
1390 : ( m_data->skinStates & ~stateFlag );
1391
1392 setSkinStates( newState );
1393}
1394
1395void QskSkinnable::replaceSkinStates(
1396 QskAspect::States newStates, int sampleIndex )
1397{
1398 /*
1399 Hack time: we might need different hints for a specific instance
1400 of a subcontrol ( f.e the selected row in a list box ), what is not
1401 supported by QskAspect.
1402
1403 As a workaround we use QskSkinStateChanger, that sets/restores this state/index
1404 while retrieving the skin hints.
1405 */
1406
1407 m_data->skinStates = newStates;
1408 m_data->sampleIndex = sampleIndex; // needed to find specific animators
1409}
1410
1411void QskSkinnable::addSkinStates( QskAspect::States states )
1412{
1413 setSkinStates( m_data->skinStates | states );
1414}
1415
1416void QskSkinnable::clearSkinStates( QskAspect::States states )
1417{
1418 setSkinStates( m_data->skinStates & ~states );
1419}
1420
1421void QskSkinnable::setSkinStates( QskAspect::States newStates )
1422{
1423 if ( m_data->skinStates == newStates )
1424 return;
1425
1426 auto item = owningItem();
1427
1428#if DEBUG_STATE
1429 const auto className = item ? item->className() : "QskSkinnable";
1430
1431 qDebug() << className << ":"
1432 << skinStateAsPrintable( m_data->skinState ) << "->"
1433 << skinStateAsPrintable( newState );
1434#endif
1435
1436 if ( item && item->window() )
1437 {
1438 if ( const auto skin = effectiveSkin() )
1439 {
1440 const auto mask = m_data->hintTable.states() | skin->hintTable().states();
1441 if ( ( newStates & mask ) != ( m_data->skinStates & mask ) )
1442 {
1443 /*
1444 When there are no aspects for the changed state bits we know
1445 that there won't be any animated transitions
1446 */
1447
1448 startHintTransitions( m_data->skinStates, newStates );
1449 }
1450 }
1451
1452 if ( item->flags() & QQuickItem::ItemHasContents )
1453 item->update();
1454 }
1455
1456 m_data->skinStates = newStates;
1457}
1458
1459bool QskSkinnable::startHintTransitions(
1460 QskAspect::States oldStates, QskAspect::States newStates, int index )
1461{
1462 QVector< QskAspect::Subcontrol > subControls;
1463
1464 if ( const auto control = qskControlCast( owningItem() ) )
1465 subControls = control->subControls();
1466
1467 return startHintTransitions( subControls, oldStates, newStates, index );
1468}
1469
1470bool QskSkinnable::startHintTransitions(
1471 const QVector< QskAspect::Subcontrol >& subControls,
1472 QskAspect::States oldStates, QskAspect::States newStates, int index )
1473{
1474 if ( !isTransitionAccepted( QskAspect() ) || subControls.isEmpty() )
1475 {
1476 // the control does not like any animated transition at the moment
1477 return false;
1478 }
1479
1480 bool started = false; // at least one transition has been started
1481
1482 QskAspect aspect;
1483 aspect.setVariation( effectiveVariation() );
1484 aspect.setSection( section() );
1485
1486 const auto skin = effectiveSkin();
1487 const auto control = qskControlCast( owningItem() );
1488 if ( control == nullptr )
1489 return false;
1490
1491 const auto primitiveCount = QskAspect::primitiveCount();
1492
1493 for ( const auto subControl : subControls )
1494 {
1495 aspect.setSubcontrol( subControl );
1496
1497 const auto& skinTable = skin->hintTable();
1498
1499 for ( uint i = 0; i < QskAspect::typeCount; i++ )
1500 {
1501 const auto type = static_cast< QskAspect::Type >( i );
1502
1503 const auto hint = effectiveAnimation( type, subControl, newStates );
1504
1505 if ( hint.duration > 0 )
1506 {
1507 /*
1508 Starting an animator for all primitives,
1509 that differ between the states
1510 */
1511
1512 for ( uint j = 0; j < primitiveCount; j++ )
1513 {
1514 const auto primitive = static_cast< QskAspect::Primitive >( j );
1515 aspect.setPrimitive( type, primitive );
1516
1517 const auto a1 = aspect | oldStates;
1518 const auto a2 = aspect | newStates;
1519
1520 bool doTransition = true;
1521
1522 if ( m_data->hintTable.states() == QskAspect::NoState )
1523 {
1524 /*
1525 In case we have no state aware aspects in the local
1526 table we can avoid starting animators for aspects,
1527 that are finally resolved from the same hint in
1528 the skin table.
1529 */
1530
1531 doTransition = !skinTable.isResolutionMatching( a1, a2 );
1532 }
1533
1534 if ( doTransition )
1535 {
1536 startHintTransition( aspect, index, hint,
1537 storedHint( a1 ), storedHint( a2 ) );
1538
1539 started = true;
1540 }
1541 }
1542 }
1543 }
1544 }
1545
1546 return started;
1547}
1548
1550{
1551 QskSkin* skin = nullptr;
1552
1553 if ( m_data->skinlet )
1554 skin = m_data->skinlet->skin();
1555
1556 if ( skin == nullptr )
1557 {
1558 if ( const auto item = owningItem() )
1559 skin = qskEffectiveSkin( item->window() );
1560 }
1561
1562 return skin ? skin : qskSkinManager->skin();
1563}
1564
1569
1570QskAspect::Section QskSkinnable::section() const
1571{
1572 return QskAspect::Body;
1573}
1574
1575void QskSkinnable::updateNode( QSGNode* parentNode )
1576{
1577 effectiveSkinlet()->updateNode( this, parentNode );
1578}
1579
1581 QskAspect::Subcontrol subControl ) const
1582{
1583 if ( const auto proxies = m_data->subcontrolProxies )
1584 {
1585 auto it = proxies->find( subControl );
1586 if ( it != proxies->end() )
1587 return it->second;
1588 }
1589
1590 return substitutedSubcontrol( subControl );
1591}
1592
1593QskAspect::Subcontrol QskSkinnable::substitutedSubcontrol(
1594 QskAspect::Subcontrol subControl ) const
1595{
1596 // derived classes might want to redirect a sub-control
1597 return subControl;
1598}
1599
1601{
1602 return qskIsControl( this )
1603 ? static_cast< QskControl* >( this ) : nullptr;
1604}
1605
1607{
1608 return qskIsControl( this )
1609 ? static_cast< const QskControl* >( this ) : nullptr;
1610}
1611
1612void QskSkinnable::debug( QDebug debug, QskAspect aspect ) const
1613{
1614 qskDebugAspect( debug, metaObject(), aspect );
1615}
1616
1617void QskSkinnable::debug( QDebug debug, QskAspect::State state ) const
1618{
1619 qskDebugStates( debug, metaObject(), state );
1620}
1621
1622void QskSkinnable::debug( QskAspect aspect ) const
1623{
1624 qskDebugAspect( qDebug(), metaObject(), aspect );
1625}
1626
1628{
1629 qskDebugStates( qDebug(), metaObject(), state );
1630}
1631
1632#ifndef QT_NO_DEBUG_STREAM
1633
1634#include <qdebug.h>
1635
1636QDebug operator<<( QDebug debug, const QskSkinHintStatus& status )
1637{
1638 QDebugStateSaver saver( debug );
1639 debug.nospace();
1640
1641 switch( status.source )
1642 {
1644 debug << "Skinnable";
1645 break;
1647 debug << "Skin";
1648 break;
1650 debug << "Animator";
1651 break;
1652 default:
1653 debug << "None";
1654 break;
1655 }
1656
1657 debug << ": " << status.aspect;
1658
1659 return debug;
1660}
1661
1662#endif
Lookup key for a QskSkinHintTable.
Definition QskAspect.h:15
State topState() const noexcept
void clearStates(States=AllStates) noexcept
Definition QskAspect.h:492
constexpr bool hasStates() const noexcept
Definition QskAspect.h:482
constexpr bool isAnimator() const noexcept
Definition QskAspect.h:407
void setStates(States) noexcept
Definition QskAspect.h:472
Variation
Some sort of variation.
Definition QskAspect.h:82
@ NoVariation
Definition QskAspect.h:83
constexpr Primitive primitive() const noexcept
Definition QskAspect.h:497
constexpr Subcontrol subControl() const noexcept
Definition QskAspect.h:417
constexpr Variation variation() const noexcept
Definition QskAspect.h:525
constexpr Type type() const noexcept
Definition QskAspect.h:447
void setAnimator(bool on) noexcept
Definition QskAspect.h:412
constexpr States states() const noexcept
Definition QskAspect.h:467
Primitive
Represents a specific element or attribute.
Definition QskAspect.h:48
@ GraphicRole
Definition QskAspect.h:56
void setPrimitive(Type, Primitive primitive) noexcept
Definition QskAspect.h:502
void setVariation(Variation) noexcept
Definition QskAspect.h:530
Type
Represents the type of the Aspect.
Definition QskAspect.h:21
Subcontrol
For use within the rendering or lay-outing of a specific QskSkinnable.
Definition QskAspect.h:104
void clearState(State) noexcept
Definition QskAspect.h:487
constexpr QskAspect stateless() const noexcept
Definition QskAspect.h:390
constexpr Primitive metricPrimitive() const noexcept
Definition QskAspect.h:514
Base class of all controls.
Definition QskControl.h:23
A paint device for scalable graphics.
Definition QskGraphic.h:28
bool isValid() const
Describes the rendering interface of a QskControl. Change the skinlet to change the appearance of the...
Definition QskSkinlet.h:34
QVariant effectiveSkinHint(QskAspect, QskSkinHintStatus *=nullptr) const
Find the value for a specific aspect.
bool resetGraphicRoleHint(QskAspect)
Removes a graphic role hint from the local table.
qreal spacingHint(QskAspect, QskSkinHintStatus *=nullptr) const
Retrieves a spacing hint.
QRectF subControlRect(const QRectF &, QskAspect::Subcontrol) const
Calculate position and size of a subControl.
QMarginsF paddingHint(QskAspect, QskSkinHintStatus *=nullptr) const
Retrieves a padding hint.
virtual ~QskSkinnable()
bool resetColor(QskAspect)
Removes a color hint from the local table.
bool resetSkinHint(QskAspect)
Remove a hint from the local hint table.
bool setAlignmentHint(QskAspect, Qt::Alignment)
Sets an alignment hint.
bool setStrutSizeHint(QskAspect, const QSizeF &)
Sets a metric hint.
bool resetFontRoleHint(QskAspect)
Removes a font role hint from the local table.
QMarginsF marginHint(QskAspect, QskSkinHintStatus *=nullptr) const
Retrieves a margin hint.
QskControl * controlCast()
void debug(QskAspect) const
QskGradient gradientHint(QskAspect, QskSkinHintStatus *=nullptr) const
Retrieves a color hint as gradient.
QSizeF innerBoxSize(QskAspect, const QSizeF &outerBoxSize) const
Calculate the size, with paddings, indentations being subtracted.
bool setBoxBorderColorsHint(QskAspect, const QskBoxBorderColors &)
Sets a border colors hint.
QskAnimationHint effectiveAnimation(QskAspect::Type, QskAspect::Subcontrol, QskAspect::States, QskSkinHintStatus *status=nullptr) const
bool setPaddingHint(QskAspect, qreal)
Sets a padding hint.
bool resetBoxBorderColorsHint(QskAspect)
Removes a border colors hint from the local table.
bool setColor(QskAspect, Qt::GlobalColor)
Sets a color hint.
bool setGradientHint(QskAspect, const QskGradient &)
Sets a gradient as color hint.
bool setSkinHint(QskAspect, const QVariant &)
Insert a hint into the local hint table.
QMarginsF innerPadding(QskAspect, const QSizeF &) const
Calculate the padding from attributes for the given aspect.
virtual bool isTransitionAccepted(QskAspect) const
Additional check if an transition should be started.
bool setBoxBorderMetricsHint(QskAspect, const QskBoxBorderMetrics &)
Sets a border metrics hint.
QFont effectiveFont(QskAspect) const
QSizeF strutSizeHint(QskAspect, QskSkinHintStatus *=nullptr) const
Retrieves a strut size hint.
bool resetBoxBorderMetricsHint(QskAspect)
Removes a border metrics hint from the local table.
QSizeF outerBoxSize(QskAspect, const QSizeF &innerBoxSize) const
Calculate the size, when being expanded by paddings, indentations.
virtual const QMetaObject * metaObject() const =0
const QskSkinlet * skinlet() const
bool setAnimationHint(QskAspect, QskAnimationHint)
const QskSkinHintTable & hintTable() const
Accessor for local skin hint table.
QskBoxShapeMetrics boxShapeHint(QskAspect, QskSkinHintStatus *=nullptr) const
Retrieves a shape hint.
bool setFontRoleHint(QskAspect, const QskFontRole &)
Sets a font role hint.
QskSkin * effectiveSkin() const
QskSkinHintStatus hintStatus(QskAspect) const
QskAnimationHint animationHint(QskAspect, QskSkinHintStatus *=nullptr) const
const QskSkinlet * effectiveSkinlet() const
QskBoxBorderColors boxBorderColorsHint(QskAspect, QskSkinHintStatus *=nullptr) const
Retrieves border colors hint.
bool setMarginHint(QskAspect, qreal)
Sets a margin hint.
bool setGraphicRoleHint(QskAspect, int role)
Sets a graphic role hint.
bool setSpacingHint(QskAspect, qreal)
Sets a spacing hint.
virtual QskAspect::Variation effectiveVariation() const
QRectF outerBox(QskAspect, const QRectF &innerBox) const
Calculate the rectangle, when being expanded by paddings, indentations.
void setSkinlet(const QskSkinlet *)
Set an individual skinlet to render/display the content.
QskFontRole fontRoleHint(QskAspect, QskSkinHintStatus *=nullptr) const
Retrieves a font role hint.
void setSkinStateFlag(QskAspect::State, bool on=true)
bool setFlagHint(QskAspect, int flag)
Sets a flag hint.
int graphicRoleHint(QskAspect, QskSkinHintStatus *=nullptr) const
A Retrieves a graphic role hint.
bool resetAlignmentHint(QskAspect)
Removes an alignment hint from the local table.
QRectF subControlContentsRect(const QRectF &, QskAspect::Subcontrol) const
Calculate the inner rectangle for subControl.
QskAspect::Subcontrol effectiveSubcontrol(QskAspect::Subcontrol) const
bool setBoxShapeHint(QskAspect, const QskBoxShapeMetrics &)
Sets a shape hint.
bool resetStrutSizeHint(QskAspect)
Removes a strut size hint from the local table.
QskColorFilter effectiveGraphicFilter(QskAspect::Subcontrol) const
QColor color(QskAspect, QskSkinHintStatus *=nullptr) const
Retrieves a color hint.
QRectF innerBox(QskAspect, const QRectF &outerBox) const
Calculate the rectangle, whith paddings, indentations being subtracted.
QskBoxBorderMetrics boxBorderMetricsHint(QskAspect, QskSkinHintStatus *=nullptr) const
Retrieves a border hint.
bool resetPaddingHint(QskAspect)
Removes a padding hint from the local table.
virtual void updateNode(QSGNode *)
bool setMetric(QskAspect, qreal)
Sets a metric hint.
bool resetBoxShapeHint(QskAspect)
Removes a shape hint from the local table.
bool resetSpacingHint(QskAspect)
Removes a spacing hint from the local table.
bool resetMetric(QskAspect)
Removes a metric hint from the local table.
bool resetMarginHint(QskAspect)
Removes a margin hint from the local table.