QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskAbstractTextInput.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskAbstractTextInput.h"
7#include "QskFontRole.h"
8#include "QskQuick.h"
9#include "QskEvent.h"
10#include "QskInternalMacros.h"
11
12#include <qguiapplication.h>
13#include <qstylehints.h>
14
15QSK_QT_PRIVATE_BEGIN
16#include <private/qquicktextedit_p.h>
17#include <private/qquicktextinput_p.h>
18
19#if QT_VERSION >= QT_VERSION_CHECK( 6, 0, 0 )
20#include <qwindow.h>
21#include <private/qeventpoint_p.h>
22#endif
23
24QSK_QT_PRIVATE_END
25
26QSK_SUBCONTROL( QskAbstractTextInput, Text )
27QSK_SUBCONTROL( QskAbstractTextInput, TextPanel )
28
29QSK_SYSTEM_STATE( QskAbstractTextInput, ReadOnly, QskAspect::FirstSystemState << 1 )
30QSK_SYSTEM_STATE( QskAbstractTextInput, Editing, QskAspect::FirstSystemState << 2 )
31QSK_SYSTEM_STATE( QskAbstractTextInput, Selected, QskAspect::FirstSystemState << 3 )
32
33/*
34 QQuickText-Edit/Input are beasts of several thousands lines of code,
35 we can't ( and don't want to ) reimplement them.
36 Instead we implement wrappers with some extra functionality to
37 have it in line with the QSkinny framework.
38
39 For some reason Qt development decided not to introduce a common
40 base class for QQuickText-Edit/input and implemented large parts
41 of the API twice. To avoid that we also have to copy those parts to our
42 wrappers we need the ugly implementation you find in this file.
43 */
44
45static inline QVariant qskInputMethodQuery(
46 const QQuickItem* item, Qt::InputMethodQuery query, QVariant argument )
47{
48 if ( auto input = qobject_cast< const QQuickTextInput* >( item ) )
49 return input->inputMethodQuery( query, argument );
50
51 if ( auto edit = qobject_cast< const QQuickTextEdit* >( item ) )
52 return edit->inputMethodQuery( query, argument );
53
54 return QVariant();
55}
56
57inline void qskSetAlignment( QQuickItem* item, Qt::Alignment alignment )
58{
59 item->setProperty( "horizontalAlignment", int( alignment ) & 0x0f );
60 item->setProperty( "verticalAlignment", int( alignment ) & 0xf0 );
61}
62
63static inline void qskTranslateMouseEventPosition(
64 QMouseEvent* mouseEvent, const QPointF& offset )
65{
66#if QT_VERSION >= QT_VERSION_CHECK( 6, 0, 0 )
67 auto& point = mouseEvent->point(0);
68
69 QMutableEventPoint::setPosition(
70 point, point.position() + offset );
71#else
72 mouseEvent->setLocalPos( mouseEvent->localPos() + offset );
73#endif
74}
75
76static inline void qskForwardEvent( QQuickItem* item, QEvent* event )
77{
78 switch( static_cast< int >( event->type() ) )
79 {
80 case QEvent::MouseButtonDblClick:
81 case QEvent::MouseButtonPress:
82 case QEvent::MouseButtonRelease:
83 case QEvent::MouseMove:
84 {
85 const auto pos = item->position();
86
87 auto mouseEvent = static_cast< QMouseEvent* >( event );
88
89 /*
90 As the event was sent for the parent item
91 we have to translate the position into
92 our coordinate system.
93 */
94 qskTranslateMouseEventPosition( mouseEvent, -pos );
95
96 QMetaObject::invokeMethod( item, "handleEvent",
97 Qt::DirectConnection, Q_ARG( QEvent*, event ) );
98
99 qskTranslateMouseEventPosition( mouseEvent, pos );
100
101 break;
102 }
103 default:
104 QMetaObject::invokeMethod( item, "handleEvent",
105 Qt::DirectConnection, Q_ARG( QEvent*, event ) );
106 }
107}
108
109class QskAbstractTextInput::PrivateData
110{
111 public:
112 ActivationModes activationModes;
113
114 QQuickItem* input = nullptr;
115 QQuickTextInput* textInput = nullptr;
116 QQuickTextEdit* textEdit = nullptr;
117};
118
119#define INPUT_INVOKE(func) \
120 ( m_data->textInput ? m_data->textInput->func() : m_data->textEdit->func() )
121
122#define INPUT_INVOKE_ARG(func, arg) \
123 ( m_data->textInput ? m_data->textInput->func( arg ) : m_data->textEdit->func( arg ) )
124
125#define INPUT_CONNECT2( func1, func2 ) \
126 m_data->textInput \
127 ? connect( m_data->textInput, &QQuickTextInput::func1, this, &QskAbstractTextInput::func2 ) \
128 : connect( m_data->textEdit, &QQuickTextEdit::func1, this, &QskAbstractTextInput::func2 )
129
130#define INPUT_CONNECT1( func ) INPUT_CONNECT2( func, func )
131
132#define INPUT_CONNECT_ARG( func, get ) \
133 do \
134 { \
135 auto f = [this]() { Q_EMIT func( get() ); }; \
136 m_data->textInput \
137 ? connect( m_data->textInput, &QQuickTextInput::func, this, f ) \
138 : connect( m_data->textEdit, &QQuickTextEdit::func, this, f ); \
139 } while ( false )
140
141QskAbstractTextInput::QskAbstractTextInput( QQuickItem* parent )
142 : Inherited( parent )
143 , m_data( new PrivateData() )
144{
145 m_data->activationModes = ActivationOnMouse | ActivationOnKey;
146
147 setPolishOnResize( true );
148
149 setAcceptHoverEvents( true );
150 setFocusPolicy( Qt::StrongFocus );
151
152 setFlag( QQuickItem::ItemAcceptsInputMethod );
153}
154
155QskAbstractTextInput::~QskAbstractTextInput()
156{
157}
158
159void QskAbstractTextInput::setup( QQuickItem* wrappedInput )
160{
161 m_data->input = wrappedInput;
162
163 m_data->textInput = qobject_cast< QQuickTextInput* >( wrappedInput );
164 m_data->textEdit = qobject_cast< QQuickTextEdit* >( wrappedInput );
165
166 INPUT_CONNECT1( textChanged );
167 INPUT_CONNECT1( preeditTextChanged );
168 INPUT_CONNECT1( selectedTextChanged );
169 INPUT_CONNECT1( readOnlyChanged );
170 INPUT_CONNECT1( overwriteModeChanged );
171 INPUT_CONNECT1( cursorVisibleChanged );
172 INPUT_CONNECT_ARG( cursorPositionChanged, cursorPosition );
173 INPUT_CONNECT1( selectByMouseChanged );
174 INPUT_CONNECT_ARG( persistentSelectionChanged, persistentSelection );
175 INPUT_CONNECT_ARG( wrapModeChanged, wrapMode );
176 INPUT_CONNECT2( contentSizeChanged, resetImplicitSize );
177
178 INPUT_CONNECT_ARG( canUndoChanged, canUndo );
179 INPUT_CONNECT_ARG( canRedoChanged, canRedo );
180 INPUT_CONNECT_ARG( canPasteChanged, canPaste );
181
182 INPUT_CONNECT_ARG( inputMethodHintsChanged, inputMethodHints );
183 INPUT_CONNECT_ARG( inputMethodComposingChanged, isInputMethodComposing );
184
185 /*
186 Other properties offered from QQuickTextInput/QQuickTextEdit:
187
188 - cursorRectangleChanged
189 - cursorDelegateChanged
190
191 The default implementation creates a QQuickRectangle for the cursor
192 that can be replaced by a delegate.
193
194 However the concept of QSkinny would be to customize the
195 the cursor using skin hints and/or creating the scene graph node
196 from the skinlet. TODO ...
197
198 - selectionColorChanged
199 - selectedTextColorChanged
200
201 This properties are set from the skin hints and are not worth to appear
202 at the public API
203
204 - horizontalAlignmentChanged
205 - verticalAlignmentChanged
206 - effectiveHorizontalAlignmentChanged
207
208 covered by QskAbstractTextInput::alignmentChanged
209
210 - activeFocusOnPressChanged
211
212 covered by QskAbstractTextInput::activationModesChanged
213
214 - selectionStartChanged;
215 - selectionEndChanged;
216
217 Do we need this ?
218
219 - mouseSelectionModeChanged
220
221 What to do with this mode. TODO ...
222
223 - editingFinished
224
225 This signal should never be emitted as it happens on
226 events ( focusOut, commit keys ) that are handled in
227 QskAbstractTextInput and are indicated with editingChanged( bool );
228 ( Maybe having an assertion TODO ... )
229
230 - contentSizeChanged
231
232 Need to understand what this information might be good for.
233 Maybe the sizeHints ....
234
235 - renderTypeChanged
236
237 This one is about the type of renderer to be used. This is similar
238 to QskItem::PreferRasterForTextures and instead of having a flag in
239 QskAbstractTextInput we might want to have a more general solution
240 in QskItem. TODO ...
241 */
242}
243
244QskAbstractTextInput::ActivationModes QskAbstractTextInput::activationModes() const
245{
246 return m_data->activationModes;
247}
248
249void QskAbstractTextInput::setActivationModes( ActivationModes modes )
250{
251 if ( m_data->activationModes != modes )
252 {
253 m_data->activationModes = modes;
254 Q_EMIT activationModesChanged();
255 }
256}
257
258#if 1
259
260// stupid code forwarding calls 1:1 to the wrapped item
261
262bool QskAbstractTextInput::selectByMouse() const
263 { return INPUT_INVOKE( selectByMouse ); }
264
265void QskAbstractTextInput::setSelectByMouse( bool on )
266 { INPUT_INVOKE_ARG( setSelectByMouse, on ); }
267
268bool QskAbstractTextInput::persistentSelection() const
269 { return INPUT_INVOKE( persistentSelection ); }
270
271void QskAbstractTextInput::setPersistentSelection( bool on )
272 { INPUT_INVOKE_ARG( setPersistentSelection, on ); }
273
274int QskAbstractTextInput::length() const
275 { return INPUT_INVOKE( length ); }
276
277QString QskAbstractTextInput::text() const
278 { return INPUT_INVOKE( text ); }
279
280void QskAbstractTextInput::setText( const QString& text )
281 { INPUT_INVOKE_ARG( setText, text ); }
282
283QString QskAbstractTextInput::preeditText() const
284 { return INPUT_INVOKE( preeditText ); }
285
286QString QskAbstractTextInput::selectedText() const
287 { return INPUT_INVOKE( selectedText ); }
288
289bool QskAbstractTextInput::isInputMethodComposing() const
290 { return INPUT_INVOKE( isInputMethodComposing ); }
291
292bool QskAbstractTextInput::overwriteMode() const
293 { return INPUT_INVOKE( overwriteMode ); }
294
295void QskAbstractTextInput::setOverwriteMode( bool on )
296 { INPUT_INVOKE_ARG( setOverwriteMode, on ); }
297
298int QskAbstractTextInput::cursorPosition() const
299 { return INPUT_INVOKE( cursorPosition ); }
300
301void QskAbstractTextInput::setCursorPosition( int pos )
302 { INPUT_INVOKE_ARG( setCursorPosition, pos ); }
303
304bool QskAbstractTextInput::isCursorVisible() const
305 { return INPUT_INVOKE( isCursorVisible ); }
306
307void QskAbstractTextInput::setCursorVisible( bool on )
308 { INPUT_INVOKE_ARG( setCursorVisible, on ); }
309
310bool QskAbstractTextInput::isReadOnly() const { return INPUT_INVOKE( isReadOnly ); }
311bool QskAbstractTextInput::canUndo() const { return INPUT_INVOKE( canUndo ); }
312bool QskAbstractTextInput::canRedo() const { return INPUT_INVOKE( canRedo ); }
313bool QskAbstractTextInput::canPaste() const { return INPUT_INVOKE( canPaste ); }
314
315void QskAbstractTextInput::clear() { INPUT_INVOKE( clear ); }
316void QskAbstractTextInput::selectAll() { INPUT_INVOKE( selectAll ); }
317void QskAbstractTextInput::deselect() { INPUT_INVOKE( deselect ); }
318void QskAbstractTextInput::cut() { INPUT_INVOKE( cut ); }
319void QskAbstractTextInput::copy() { INPUT_INVOKE( copy ); }
320void QskAbstractTextInput::paste() { INPUT_INVOKE( paste ); }
321void QskAbstractTextInput::undo() { INPUT_INVOKE( undo ); }
322void QskAbstractTextInput::redo() { INPUT_INVOKE( redo ); }
323
324#endif
325
326bool QskAbstractTextInput::hasSelectedText() const
327{
328 return INPUT_INVOKE( selectionEnd ) > INPUT_INVOKE( selectionStart );
329}
330
331
332void QskAbstractTextInput::setFontRole( const QskFontRole& role )
333{
334 if ( setFontRoleHint( Text, role ) )
335 {
336 const auto queries = Qt::ImCursorRectangle | Qt::ImFont | Qt::ImAnchorRectangle;
337 qskUpdateInputMethod( this, queries );
338
339 Q_EMIT fontRoleChanged( role );
340 }
341}
342
343void QskAbstractTextInput::resetFontRole()
344{
345 if ( resetFontRoleHint( Text ) )
346 {
347 const auto queries = Qt::ImCursorRectangle | Qt::ImFont | Qt::ImAnchorRectangle;
348 qskUpdateInputMethod( this, queries );
349
350 Q_EMIT fontRoleChanged( fontRole() );
351 }
352}
353
354QskFontRole QskAbstractTextInput::fontRole() const
355{
356 return fontRoleHint( Text );
357}
358
359QFont QskAbstractTextInput::font() const
360{
361 return effectiveFont( Text );
362}
363
364QVariant QskAbstractTextInput::inputMethodQuery(
365 Qt::InputMethodQuery query ) const
366{
367 return inputMethodQuery( query, QVariant() );
368}
369
370QVariant QskAbstractTextInput::inputMethodQuery(
371 Qt::InputMethodQuery query, const QVariant& argument ) const
372{
373 switch ( query )
374 {
375 case Qt::ImEnabled:
376 {
377 return QVariant( ( bool ) ( flags() & ItemAcceptsInputMethod ) );
378 }
379 case Qt::ImFont:
380 {
381 return font();
382 }
383 case Qt::ImPreferredLanguage:
384 {
385 return locale();
386 }
387 case Qt::ImInputItemClipRectangle:
388 case Qt::ImCursorRectangle:
389 {
390 QVariant v = qskInputMethodQuery( m_data->input, query, argument );
391#if 1
392 if ( v.canConvert< QRectF >() )
393 v.setValue( v.toRectF().translated( m_data->input->position() ) );
394#endif
395 return v;
396 }
397 default:
398 {
399 return qskInputMethodQuery( m_data->input, query, argument );
400 }
401 }
402}
403
404Qt::InputMethodHints QskAbstractTextInput::inputMethodHints() const
405{
406 return INPUT_INVOKE( inputMethodHints );
407}
408
409void QskAbstractTextInput::setInputMethodHints( Qt::InputMethodHints hints )
410{
411 if ( inputMethodHints() != hints )
412 {
413 INPUT_INVOKE_ARG( setInputMethodHints, hints );
414 qskUpdateInputMethod( this, Qt::ImHints );
415 }
416}
417
418bool QskAbstractTextInput::event( QEvent* event )
419{
420 if ( event->type() == QEvent::LocaleChange )
421 qskUpdateInputMethod( this, Qt::ImPreferredLanguage );
422
423 if ( event->type() == QEvent::ShortcutOverride )
424 {
425 forwardEvent( event );
426 return event->isAccepted();
427 }
428
429 return Inherited::event( event );
430}
431
432void QskAbstractTextInput::mousePressEvent( QMouseEvent* event )
433{
434 forwardEvent( event );
435
436 if ( !isReadOnly() && !qGuiApp->styleHints()->setFocusOnTouchRelease() )
437 setEditing( true );
438}
439
440void QskAbstractTextInput::mouseMoveEvent( QMouseEvent* event )
441{
442 forwardEvent( event );
443}
444
445void QskAbstractTextInput::mouseReleaseEvent( QMouseEvent* event )
446{
447 forwardEvent( event );
448
449 if ( !isReadOnly() && qGuiApp->styleHints()->setFocusOnTouchRelease() )
450 setEditing( true );
451}
452
453void QskAbstractTextInput::mouseDoubleClickEvent( QMouseEvent* event )
454{
455 forwardEvent( event );
456}
457
458void QskAbstractTextInput::keyPressEvent( QKeyEvent* event )
459{
460 if ( isEditing() )
461 {
462 switch ( event->key() )
463 {
464 case Qt::Key_Escape:
465 {
466 setEditing( false );
467 break;
468 }
469
470 case Qt::Key_Enter:
471 case Qt::Key_Return:
472 {
473 const auto hints = inputMethodQuery( Qt::ImHints ).toInt();
474 if ( !( hints & Qt::ImhMultiLine ) )
475 {
476 if ( auto input = m_data->textInput )
477 {
478 bool accept = input->hasAcceptableInput();
479 if ( !accept )
480 {
481 QMetaObject::invokeMethod( input, "fixup",
482 Qt::DirectConnection, Q_RETURN_ARG(bool, accept) );
483 }
484
485 if ( !accept )
486 return; // ignore
487 }
488
489 QGuiApplication::inputMethod()->commit();
490 setEditing( false );
491 }
492
493 break;
494 }
495 }
496
497 if ( isEditing() )
498 {
499 forwardEvent( event );
500 }
501 else
502 {
503 // When returning from a virtual keyboard
504 qskForceActiveFocus( this, Qt::PopupFocusReason );
505 }
506 }
507 else
508 {
509 if ( ( activationModes() & ActivationOnKey ) && !event->isAutoRepeat() )
510 {
511 if ( event->key() == Qt::Key_Select || event->key() == Qt::Key_Space )
512 {
513 setEditing( true );
514 return;
515 }
516 }
517
518 Inherited::keyPressEvent( event );
519 }
520}
521
522void QskAbstractTextInput::keyReleaseEvent( QKeyEvent* event )
523{
524 Inherited::keyReleaseEvent( event );
525}
526
527void QskAbstractTextInput::focusInEvent( QFocusEvent* event )
528{
529 if ( activationModes() & ActivationOnFocus )
530 {
531 switch ( event->reason() )
532 {
533 case Qt::ActiveWindowFocusReason:
534 case Qt::PopupFocusReason:
535 break;
536
537 default:
538#if 1
539 // auto selecting the complete text ???
540#endif
541 setEditing( true );
542 }
543 }
544
545 Inherited::focusInEvent( event );
546}
547
548void QskAbstractTextInput::focusOutEvent( QFocusEvent* event )
549{
550 switch ( event->reason() )
551 {
552 case Qt::ActiveWindowFocusReason:
553 case Qt::PopupFocusReason:
554 {
555 break;
556 }
557 default:
558 {
559 deselect();
560 setEditing( false );
561 }
562 }
563
564 Inherited::focusOutEvent( event );
565}
566
567void QskAbstractTextInput::inputMethodEvent( QInputMethodEvent* event )
568{
569 const bool hadCursor = isCursorVisible();
570
571 forwardEvent( event );
572
573 if ( isCursorVisible() && !hadCursor )
574 {
575 /*
576 The initial InputMethod events might be sent from the
577 platform depending on focus. Unfortunately an
578 empty dummy event ( = no attributes ) leads to showing
579 the cursor.
580 */
581 if ( isEditing() )
582 setCursorVisible( false );
583 }
584}
585
586void QskAbstractTextInput::setReadOnly( bool on )
587{
588 if ( on == isReadOnly() )
589 return;
590
591#if 1
592 // do we want to be able to restore the previous policy ?
593 setFocusPolicy( Qt::NoFocus );
594#endif
595
596 INPUT_INVOKE_ARG( setReadOnly, on );
597
598 // we are killing user settings here ?
599 m_data->input->setFlag( QQuickItem::ItemAcceptsInputMethod, !on );
600
601#if QT_VERSION >= QT_VERSION_CHECK( 6, 2, 0 )
602 qskUpdateInputMethod( this, Qt::ImReadOnly );
603#else
604 qskUpdateInputMethod( this, Qt::ImEnabled );
605#endif
606
607 setSkinStateFlag( ReadOnly, on );
608}
609
610bool QskAbstractTextInput::isEditing() const
611{
612 return hasSkinState( Editing );
613}
614
615void QskAbstractTextInput::setEditing( bool on )
616{
617 if ( ( on == isEditing() ) || ( on && isReadOnly() ) )
618 return;
619
620 setSkinStateFlag( Editing, on );
621
622 QMetaObject::invokeMethod( m_data->input, "setEditing",
623 Qt::DirectConnection, Q_ARG( bool, on ) );
624
625 qskInputMethodSetVisible( this, on );
626 Q_EMIT editingChanged( on );
627}
628
629void QskAbstractTextInput::setWrapMode( QskTextOptions::WrapMode wrapMode )
630{
631 if ( m_data->textInput )
632 {
633 auto mode = static_cast< QQuickTextInput::WrapMode >( wrapMode );
634 m_data->textInput->setWrapMode( mode );
635 }
636 else
637 {
638 auto mode = static_cast< QQuickTextEdit::WrapMode >( wrapMode );
639 m_data->textEdit->setWrapMode( mode );
640 }
641}
642
643QskTextOptions::WrapMode QskAbstractTextInput::wrapMode() const
644{
645 if ( m_data->textInput )
646 return static_cast< QskTextOptions::WrapMode >( m_data->textInput->wrapMode() );
647 else
648 return static_cast< QskTextOptions::WrapMode >( m_data->textEdit->wrapMode() );
649}
650
651QSizeF QskAbstractTextInput::unwrappedTextSize() const
652{
653 const auto w = INPUT_INVOKE( contentWidth );
654 const auto h = INPUT_INVOKE( contentHeight );
655
656 return QSizeF( w, h );
657}
658
659void QskAbstractTextInput::setTextColor( const QColor& color )
660{
661 if ( setColor( Text, color ) )
662 Q_EMIT textColorChanged( color );
663}
664
665void QskAbstractTextInput::resetTextColor()
666{
667 if ( resetColor( Text ) )
668 Q_EMIT textColorChanged( color( Text ) );
669}
670
671QColor QskAbstractTextInput::textColor() const
672{
673 return color( Text );
674}
675
676void QskAbstractTextInput::setAlignment( Qt::Alignment alignment )
677{
678 if ( setAlignmentHint( Text, alignment ) )
679 {
680 qskSetAlignment( m_data->input, alignment );
681 Q_EMIT alignmentChanged();
682 }
683}
684
685void QskAbstractTextInput::resetAlignment()
686{
687 if ( resetAlignmentHint( Text ) )
688 {
689 qskSetAlignment( m_data->input, alignment() );
690 Q_EMIT alignmentChanged();
691 }
692}
693
694Qt::Alignment QskAbstractTextInput::alignment() const
695{
696 Qt::Alignment alignment = Qt::AlignLeft;
697 if ( qobject_cast< const QQuickTextEdit* >( m_data->input ) )
698 alignment |= Qt::AlignTop;
699 else
700 alignment |= Qt::AlignVCenter;
701
702 return alignmentHint( Text, alignment );
703}
704
705void QskAbstractTextInput::updateLayout()
706{
707 QMetaObject::invokeMethod( m_data->input, "updateMetrics" );
708
709 const auto subControl = effectiveSubcontrol( Text );
710 qskSetItemGeometry( m_data->input, subControlRect( subControl ) );
711}
712
714{
715 QMetaObject::invokeMethod( m_data->input, "updateColors" );
716 Inherited::updateNode( node );
717}
718
719void QskAbstractTextInput::forwardEvent( QEvent* event )
720{
721 qskForwardEvent( m_data->input, event );
722}
723
724#include "moc_QskAbstractTextInput.cpp"
void updateNode(QSGNode *) override
Lookup key for a QskSkinHintTable.
Definition QskAspect.h:15
@ FirstSystemState
Definition QskAspect.h:115
QRectF subControlRect(QskAspect::Subcontrol) const
QLocale locale
Definition QskControl.h:27
void resetImplicitSize()
Definition QskItem.cpp:727
bool resetColor(QskAspect)
Removes a color hint from the local table.
bool setAlignmentHint(QskAspect, Qt::Alignment)
Sets an alignment hint.
bool resetFontRoleHint(QskAspect)
Removes a font role hint from the local table.
bool setColor(QskAspect, Qt::GlobalColor)
Sets a color hint.
QFont effectiveFont(QskAspect) const
bool setFontRoleHint(QskAspect, const QskFontRole &)
Sets a font role hint.
QskFontRole fontRoleHint(QskAspect, QskSkinHintStatus *=nullptr) const
Retrieves a font role hint.
void setSkinStateFlag(QskAspect::State, bool on=true)
bool resetAlignmentHint(QskAspect)
Removes an alignment hint from the local table.
QskAspect::Subcontrol effectiveSubcontrol(QskAspect::Subcontrol) const
QColor color(QskAspect, QskSkinHintStatus *=nullptr) const
Retrieves a color hint.
virtual void updateNode(QSGNode *)
Qt::Alignment alignmentHint(QskAspect, Qt::Alignment=Qt::Alignment()) const
Retrieves an alignment hint.