QAndroidPlatformInputContext: send composition text and cursor jointly

QAndroidPlatformInputContext::focusObjectStopComposing() sends an input
event for each character newly added by the Android virtual keyboard.
It then sends a second input event to notify that the cursor has
advanced to the position after the new character.
The implicit assumption is, that the receiver of the input event does
not change the text.

If e.g. QLineEdit::setText() is called in the QLineEdit::textEdited
slot, the text does change. If the change implies a cursor change,
QLineEdit notifies the platform input context about it.
However, by sending the second input event, QAndroidPlatformContent
returns the cursor back to the position after the last character added
by the virtual keyboard.

This patch joins the composed text and the cursor position into one
single input method event. A new cursor position, set by the receiver
of the input method event, is no longer overridden.
The patch adds test functionality to tst_QLineEdit::setText().

Fixes: QTBUG-115756
Pick-to: 6.6 6.5 6.2
Change-Id: I85ffac5d6bab93ccb144be0f5b8083258a270550
Reviewed-by: Tor Arne Vestbø <tor.arne.vestbo@qt.io>
This commit is contained in:
Axel Spoerl 2023-08-13 11:39:01 +02:00
parent b1c062d078
commit be3b9b2ab1
2 changed files with 42 additions and 20 deletions

View File

@ -1131,21 +1131,13 @@ bool QAndroidInputContext::focusObjectStopComposing()
m_composingCursor = -1;
{
// commit the composing test
QList<QInputMethodEvent::Attribute> attributes;
QInputMethodEvent event(QString(), attributes);
event.setCommitString(m_composingText);
sendInputMethodEvent(&event);
}
{
// Moving Qt's cursor to where the preedit cursor used to be
QList<QInputMethodEvent::Attribute> attributes;
attributes.append(
QInputMethodEvent::Attribute(QInputMethodEvent::Selection, localCursorPos, 0));
QInputMethodEvent event(QString(), attributes);
sendInputMethodEvent(&event);
}
// commit composing text and cursor position
QList<QInputMethodEvent::Attribute> attributes;
attributes.append(
QInputMethodEvent::Attribute(QInputMethodEvent::Selection, localCursorPos, 0));
QInputMethodEvent event(QString(), attributes);
event.setCommitString(m_composingText);
sendInputMethodEvent(&event);
return true;
}

View File

@ -1590,14 +1590,44 @@ void tst_QLineEdit::textMask()
QCOMPARE( testWidget->text(), insertString );
}
class LineEditChangingText : public QLineEdit
{
Q_OBJECT
public:
LineEditChangingText(QWidget *parent) : QLineEdit(parent)
{
connect(this, &QLineEdit::textEdited, this, &LineEditChangingText::onTextEdited);
}
public slots:
void onTextEdited(const QString &text)
{
if (text.length() == 3)
setText(text + "-");
}
};
void tst_QLineEdit::setText()
{
QLineEdit *testWidget = ensureTestWidget();
QSignalSpy editedSpy(testWidget, SIGNAL(textEdited(QString)));
QSignalSpy changedSpy(testWidget, SIGNAL(textChanged(QString)));
testWidget->setText("hello");
QCOMPARE(editedSpy.size(), 0);
QCOMPARE(changedSpy.value(0).value(0).toString(), QString("hello"));
{
QSignalSpy editedSpy(testWidget, &QLineEdit::textEdited);
QSignalSpy changedSpy(testWidget, &QLineEdit::textChanged);
testWidget->setText("hello");
QCOMPARE(editedSpy.size(), 0);
QCOMPARE(changedSpy.value(0).value(0).toString(), QString("hello"));
}
QTestEventList keys;
keys.addKeyClick(Qt::Key_A);
keys.addKeyClick(Qt::Key_B);
keys.addKeyClick(Qt::Key_C);
LineEditChangingText lineEdit(nullptr);
keys.simulate(&lineEdit);
QCOMPARE(lineEdit.text(), "abc-");
QCOMPARE(lineEdit.cursorPosition(), 4);
}
void tst_QLineEdit::displayText_data()