6#include "QskAbstractTextInput.h"
7#include "QskFontRole.h"
10#include "QskInternalMacros.h"
12#include <qguiapplication.h>
13#include <qstylehints.h>
16#include <private/qquicktextedit_p.h>
17#include <private/qquicktextinput_p.h>
19#if QT_VERSION >= QT_VERSION_CHECK( 6, 0, 0 )
21#include <private/qeventpoint_p.h>
45static inline QVariant qskInputMethodQuery(
46 const QQuickItem* item, Qt::InputMethodQuery query, QVariant argument )
48 if (
auto input = qobject_cast< const QQuickTextInput* >( item ) )
49 return input->inputMethodQuery( query, argument );
51 if (
auto edit = qobject_cast< const QQuickTextEdit* >( item ) )
52 return edit->inputMethodQuery( query, argument );
57inline void qskSetAlignment( QQuickItem* item, Qt::Alignment alignment )
59 item->setProperty(
"horizontalAlignment",
int( alignment ) & 0x0f );
60 item->setProperty(
"verticalAlignment",
int( alignment ) & 0xf0 );
63static inline void qskTranslateMouseEventPosition(
64 QMouseEvent* mouseEvent,
const QPointF& offset )
66#if QT_VERSION >= QT_VERSION_CHECK( 6, 0, 0 )
67 auto& point = mouseEvent->point(0);
69 QMutableEventPoint::setPosition(
70 point, point.position() + offset );
72 mouseEvent->setLocalPos( mouseEvent->localPos() + offset );
76static inline void qskForwardEvent( QQuickItem* item, QEvent* event )
78 switch(
static_cast< int >( event->type() ) )
80 case QEvent::MouseButtonDblClick:
81 case QEvent::MouseButtonPress:
82 case QEvent::MouseButtonRelease:
83 case QEvent::MouseMove:
85 const auto pos = item->position();
87 auto mouseEvent =
static_cast< QMouseEvent*
>( event );
94 qskTranslateMouseEventPosition( mouseEvent, -pos );
96 QMetaObject::invokeMethod( item,
"handleEvent",
97 Qt::DirectConnection, Q_ARG( QEvent*, event ) );
99 qskTranslateMouseEventPosition( mouseEvent, pos );
104 QMetaObject::invokeMethod( item,
"handleEvent",
105 Qt::DirectConnection, Q_ARG( QEvent*, event ) );
109class QskAbstractTextInput::PrivateData
112 ActivationModes activationModes;
114 QQuickItem* input =
nullptr;
115 QQuickTextInput* textInput =
nullptr;
116 QQuickTextEdit* textEdit =
nullptr;
119#define INPUT_INVOKE(func) \
120 ( m_data->textInput ? m_data->textInput->func() : m_data->textEdit->func() )
122#define INPUT_INVOKE_ARG(func, arg) \
123 ( m_data->textInput ? m_data->textInput->func( arg ) : m_data->textEdit->func( arg ) )
125#define INPUT_CONNECT2( func1, func2 ) \
127 ? connect( m_data->textInput, &QQuickTextInput::func1, this, &QskAbstractTextInput::func2 ) \
128 : connect( m_data->textEdit, &QQuickTextEdit::func1, this, &QskAbstractTextInput::func2 )
130#define INPUT_CONNECT1( func ) INPUT_CONNECT2( func, func )
132#define INPUT_CONNECT_ARG( func, get ) \
135 auto f = [this]() { Q_EMIT func( get() ); }; \
137 ? connect( m_data->textInput, &QQuickTextInput::func, this, f ) \
138 : connect( m_data->textEdit, &QQuickTextEdit::func, this, f ); \
141QskAbstractTextInput::QskAbstractTextInput( QQuickItem* parent )
142 : Inherited( parent )
143 , m_data( new PrivateData() )
145 m_data->activationModes = ActivationOnMouse | ActivationOnKey;
147 setPolishOnResize(
true );
149 setAcceptHoverEvents(
true );
150 setFocusPolicy( Qt::StrongFocus );
152 setFlag( QQuickItem::ItemAcceptsInputMethod );
155QskAbstractTextInput::~QskAbstractTextInput()
159void QskAbstractTextInput::setup( QQuickItem* wrappedInput )
161 m_data->input = wrappedInput;
163 m_data->textInput = qobject_cast< QQuickTextInput* >( wrappedInput );
164 m_data->textEdit = qobject_cast< QQuickTextEdit* >( wrappedInput );
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 );
178 INPUT_CONNECT_ARG( canUndoChanged, canUndo );
179 INPUT_CONNECT_ARG( canRedoChanged, canRedo );
180 INPUT_CONNECT_ARG( canPasteChanged, canPaste );
182 INPUT_CONNECT_ARG( inputMethodHintsChanged, inputMethodHints );
183 INPUT_CONNECT_ARG( inputMethodComposingChanged, isInputMethodComposing );
244QskAbstractTextInput::ActivationModes QskAbstractTextInput::activationModes()
const
246 return m_data->activationModes;
249void QskAbstractTextInput::setActivationModes( ActivationModes modes )
251 if ( m_data->activationModes != modes )
253 m_data->activationModes = modes;
254 Q_EMIT activationModesChanged();
262bool QskAbstractTextInput::selectByMouse()
const
263 {
return INPUT_INVOKE( selectByMouse ); }
265void QskAbstractTextInput::setSelectByMouse(
bool on )
266 { INPUT_INVOKE_ARG( setSelectByMouse, on ); }
268bool QskAbstractTextInput::persistentSelection()
const
269 {
return INPUT_INVOKE( persistentSelection ); }
271void QskAbstractTextInput::setPersistentSelection(
bool on )
272 { INPUT_INVOKE_ARG( setPersistentSelection, on ); }
274int QskAbstractTextInput::length()
const
275 {
return INPUT_INVOKE( length ); }
277QString QskAbstractTextInput::text()
const
278 {
return INPUT_INVOKE( text ); }
280void QskAbstractTextInput::setText(
const QString& text )
281 { INPUT_INVOKE_ARG( setText, text ); }
283QString QskAbstractTextInput::preeditText()
const
284 {
return INPUT_INVOKE( preeditText ); }
286QString QskAbstractTextInput::selectedText()
const
287 {
return INPUT_INVOKE( selectedText ); }
289bool QskAbstractTextInput::isInputMethodComposing()
const
290 {
return INPUT_INVOKE( isInputMethodComposing ); }
292bool QskAbstractTextInput::overwriteMode()
const
293 {
return INPUT_INVOKE( overwriteMode ); }
295void QskAbstractTextInput::setOverwriteMode(
bool on )
296 { INPUT_INVOKE_ARG( setOverwriteMode, on ); }
298int QskAbstractTextInput::cursorPosition()
const
299 {
return INPUT_INVOKE( cursorPosition ); }
301void QskAbstractTextInput::setCursorPosition(
int pos )
302 { INPUT_INVOKE_ARG( setCursorPosition, pos ); }
304bool QskAbstractTextInput::isCursorVisible()
const
305 {
return INPUT_INVOKE( isCursorVisible ); }
307void QskAbstractTextInput::setCursorVisible(
bool on )
308 { INPUT_INVOKE_ARG( setCursorVisible, on ); }
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 ); }
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 ); }
326bool QskAbstractTextInput::hasSelectedText()
const
328 return INPUT_INVOKE( selectionEnd ) > INPUT_INVOKE( selectionStart );
332void QskAbstractTextInput::setFontRole(
const QskFontRole& role )
336 const auto queries = Qt::ImCursorRectangle | Qt::ImFont | Qt::ImAnchorRectangle;
337 qskUpdateInputMethod(
this, queries );
339 Q_EMIT fontRoleChanged( role );
343void QskAbstractTextInput::resetFontRole()
347 const auto queries = Qt::ImCursorRectangle | Qt::ImFont | Qt::ImAnchorRectangle;
348 qskUpdateInputMethod(
this, queries );
350 Q_EMIT fontRoleChanged( fontRole() );
359QFont QskAbstractTextInput::font()
const
364QVariant QskAbstractTextInput::inputMethodQuery(
365 Qt::InputMethodQuery query )
const
367 return inputMethodQuery( query, QVariant() );
370QVariant QskAbstractTextInput::inputMethodQuery(
371 Qt::InputMethodQuery query,
const QVariant& argument )
const
377 return QVariant( (
bool ) ( flags() & ItemAcceptsInputMethod ) );
383 case Qt::ImPreferredLanguage:
387 case Qt::ImInputItemClipRectangle:
388 case Qt::ImCursorRectangle:
390 QVariant v = qskInputMethodQuery( m_data->input, query, argument );
392 if ( v.canConvert< QRectF >() )
393 v.setValue( v.toRectF().translated( m_data->input->position() ) );
399 return qskInputMethodQuery( m_data->input, query, argument );
404Qt::InputMethodHints QskAbstractTextInput::inputMethodHints()
const
406 return INPUT_INVOKE( inputMethodHints );
409void QskAbstractTextInput::setInputMethodHints( Qt::InputMethodHints hints )
411 if ( inputMethodHints() != hints )
413 INPUT_INVOKE_ARG( setInputMethodHints, hints );
414 qskUpdateInputMethod(
this, Qt::ImHints );
418bool QskAbstractTextInput::event( QEvent* event )
420 if ( event->type() == QEvent::LocaleChange )
421 qskUpdateInputMethod(
this, Qt::ImPreferredLanguage );
423 if ( event->type() == QEvent::ShortcutOverride )
425 forwardEvent( event );
426 return event->isAccepted();
429 return Inherited::event( event );
432void QskAbstractTextInput::mousePressEvent( QMouseEvent* event )
434 forwardEvent( event );
436 if ( !isReadOnly() && !qGuiApp->styleHints()->setFocusOnTouchRelease() )
440void QskAbstractTextInput::mouseMoveEvent( QMouseEvent* event )
442 forwardEvent( event );
445void QskAbstractTextInput::mouseReleaseEvent( QMouseEvent* event )
447 forwardEvent( event );
449 if ( !isReadOnly() && qGuiApp->styleHints()->setFocusOnTouchRelease() )
453void QskAbstractTextInput::mouseDoubleClickEvent( QMouseEvent* event )
455 forwardEvent( event );
458void QskAbstractTextInput::keyPressEvent( QKeyEvent* event )
462 switch ( event->key() )
473 const auto hints = inputMethodQuery( Qt::ImHints ).toInt();
474 if ( !( hints & Qt::ImhMultiLine ) )
476 if (
auto input = m_data->textInput )
478 bool accept = input->hasAcceptableInput();
481 QMetaObject::invokeMethod( input,
"fixup",
482 Qt::DirectConnection, Q_RETURN_ARG(
bool, accept) );
489 QGuiApplication::inputMethod()->commit();
499 forwardEvent( event );
504 qskForceActiveFocus(
this, Qt::PopupFocusReason );
509 if ( ( activationModes() & ActivationOnKey ) && !event->isAutoRepeat() )
511 if ( event->key() == Qt::Key_Select || event->key() == Qt::Key_Space )
518 Inherited::keyPressEvent( event );
522void QskAbstractTextInput::keyReleaseEvent( QKeyEvent* event )
524 Inherited::keyReleaseEvent( event );
527void QskAbstractTextInput::focusInEvent( QFocusEvent* event )
529 if ( activationModes() & ActivationOnFocus )
531 switch ( event->reason() )
533 case Qt::ActiveWindowFocusReason:
534 case Qt::PopupFocusReason:
545 Inherited::focusInEvent( event );
548void QskAbstractTextInput::focusOutEvent( QFocusEvent* event )
550 switch ( event->reason() )
552 case Qt::ActiveWindowFocusReason:
553 case Qt::PopupFocusReason:
564 Inherited::focusOutEvent( event );
567void QskAbstractTextInput::inputMethodEvent( QInputMethodEvent* event )
569 const bool hadCursor = isCursorVisible();
571 forwardEvent( event );
573 if ( isCursorVisible() && !hadCursor )
582 setCursorVisible(
false );
586void QskAbstractTextInput::setReadOnly(
bool on )
588 if ( on == isReadOnly() )
593 setFocusPolicy( Qt::NoFocus );
596 INPUT_INVOKE_ARG( setReadOnly, on );
599 m_data->input->setFlag( QQuickItem::ItemAcceptsInputMethod, !on );
601#if QT_VERSION >= QT_VERSION_CHECK( 6, 2, 0 )
602 qskUpdateInputMethod(
this, Qt::ImReadOnly );
604 qskUpdateInputMethod(
this, Qt::ImEnabled );
610bool QskAbstractTextInput::isEditing()
const
612 return hasSkinState( Editing );
615void QskAbstractTextInput::setEditing(
bool on )
617 if ( ( on == isEditing() ) || ( on && isReadOnly() ) )
622 QMetaObject::invokeMethod( m_data->input,
"setEditing",
623 Qt::DirectConnection, Q_ARG(
bool, on ) );
625 qskInputMethodSetVisible(
this, on );
626 Q_EMIT editingChanged( on );
629void QskAbstractTextInput::setWrapMode( QskTextOptions::WrapMode wrapMode )
631 if ( m_data->textInput )
633 auto mode =
static_cast< QQuickTextInput::WrapMode
>( wrapMode );
634 m_data->textInput->setWrapMode( mode );
638 auto mode =
static_cast< QQuickTextEdit::WrapMode
>( wrapMode );
639 m_data->textEdit->setWrapMode( mode );
643QskTextOptions::WrapMode QskAbstractTextInput::wrapMode()
const
645 if ( m_data->textInput )
646 return static_cast< QskTextOptions::WrapMode
>( m_data->textInput->wrapMode() );
648 return static_cast< QskTextOptions::WrapMode
>( m_data->textEdit->wrapMode() );
651QSizeF QskAbstractTextInput::unwrappedTextSize()
const
653 const auto w = INPUT_INVOKE( contentWidth );
654 const auto h = INPUT_INVOKE( contentHeight );
656 return QSizeF( w, h );
659void QskAbstractTextInput::setTextColor(
const QColor& color )
662 Q_EMIT textColorChanged(
color );
665void QskAbstractTextInput::resetTextColor()
668 Q_EMIT textColorChanged(
color( Text ) );
671QColor QskAbstractTextInput::textColor()
const
673 return color( Text );
676void QskAbstractTextInput::setAlignment( Qt::Alignment alignment )
680 qskSetAlignment( m_data->input, alignment );
681 Q_EMIT alignmentChanged();
685void QskAbstractTextInput::resetAlignment()
689 qskSetAlignment( m_data->input, alignment() );
690 Q_EMIT alignmentChanged();
694Qt::Alignment QskAbstractTextInput::alignment()
const
696 Qt::Alignment alignment = Qt::AlignLeft;
697 if ( qobject_cast< const QQuickTextEdit* >( m_data->input ) )
698 alignment |= Qt::AlignTop;
700 alignment |= Qt::AlignVCenter;
705void QskAbstractTextInput::updateLayout()
707 QMetaObject::invokeMethod( m_data->input,
"updateMetrics" );
710 qskSetItemGeometry( m_data->input,
subControlRect( subControl ) );
715 QMetaObject::invokeMethod( m_data->input,
"updateColors" );
719void QskAbstractTextInput::forwardEvent( QEvent* event )
721 qskForwardEvent( m_data->input, event );
724#include "moc_QskAbstractTextInput.cpp"
void updateNode(QSGNode *) override
Lookup key for a QskSkinHintTable.
QRectF subControlRect(QskAspect::Subcontrol) const
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.