/**************************************************************************** ** ** Copyright (C) 2012 Nokia Corporation and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/ ** ** This file is part of the test suite of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** GNU Lesser General Public License Usage ** This file may be used under the terms of the GNU Lesser General Public ** License version 2.1 as published by the Free Software Foundation and ** appearing in the file LICENSE.LGPL included in the packaging of this ** file. Please review the following information to ensure the GNU Lesser ** General Public License version 2.1 requirements will be met: ** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** In addition, as a special exception, Nokia gives you certain additional ** rights. These rights are described in the Nokia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU General ** Public License version 3.0 as published by the Free Software Foundation ** and appearing in the file LICENSE.GPL included in the packaging of this ** file. Please review the following information to ensure the GNU General ** Public License version 3.0 requirements will be met: ** http://www.gnu.org/copyleft/gpl.html. ** ** Other Usage ** Alternatively, this file may be used in accordance with the terms and ** conditions contained in a signed written agreement between you and Nokia. ** ** ** ** ** ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include #include #include "../../shared/testhttpserver.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "../../shared/util.h" #include "../../shared/platforminputcontext.h" #include #ifdef Q_OS_MAC #include #endif Q_DECLARE_METATYPE(QQuickTextEdit::SelectionMode) Q_DECLARE_METATYPE(Qt::Key) DEFINE_BOOL_CONFIG_OPTION(qmlDisableDistanceField, QML_DISABLE_DISTANCEFIELD) QString createExpectedFileIfNotFound(const QString& filebasename, const QImage& actual) { // XXX This will be replaced by some clever persistent platform image store. QString persistent_dir = QQmlDataTest::instance()->dataDirectory(); QString arch = "unknown-architecture"; // QTest needs to help with this. QString expectfile = persistent_dir + QDir::separator() + filebasename + "-" + arch + ".png"; if (!QFile::exists(expectfile)) { actual.save(expectfile); qWarning() << "created" << expectfile; } return expectfile; } typedef QPair Key; class tst_qquicktextedit : public QQmlDataTest { Q_OBJECT public: tst_qquicktextedit(); private slots: void cleanup(); void text(); void width(); void wrap(); void textFormat(); void alignments(); void alignments_data(); // ### these tests may be trivial void hAlign(); void hAlign_RightToLeft(); void vAlign(); void font(); void color(); void textMargin(); void persistentSelection(); void focusOnPress(); void selection(); void isRightToLeft_data(); void isRightToLeft(); void keySelection(); void moveCursorSelection_data(); void moveCursorSelection(); void moveCursorSelectionSequence_data(); void moveCursorSelectionSequence(); void mouseSelection_data(); void mouseSelection(); void mouseSelectionMode_data(); void mouseSelectionMode(); void dragMouseSelection(); void inputMethodHints(); void positionAt(); void linkActivated(); void cursorDelegate_data(); void cursorDelegate(); void cursorVisible(); void delegateLoading_data(); void delegateLoading(); void navigation(); void readOnly(); void copyAndPaste(); void canPaste(); void canPasteEmpty(); void textInput(); void inputMethodUpdate(); void openInputPanel(); void geometrySignals(); void pastingRichText_QTBUG_14003(); void implicitSize_data(); void implicitSize(); void contentSize(); void implicitSizeBinding_data(); void implicitSizeBinding(); void preeditCursorRectangle(); void inputMethodComposing(); void cursorRectangleSize(); void getText_data(); void getText(); void getFormattedText_data(); void getFormattedText(); void insert_data(); void insert(); void remove_data(); void remove(); void keySequence_data(); void keySequence(); void undo_data(); void undo(); void redo_data(); void redo(); void undo_keypressevents_data(); void undo_keypressevents(); void baseUrl(); void embeddedImages(); void embeddedImages_data(); void emptytags_QTBUG_22058(); private: void simulateKeys(QWindow *window, const QList &keys); void simulateKeys(QWindow *window, const QKeySequence &sequence); void simulateKey(QWindow *, int key, Qt::KeyboardModifiers modifiers = 0); QStringList standard; QStringList richText; QStringList hAlignmentStrings; QStringList vAlignmentStrings; QList vAlignments; QList hAlignments; QStringList colorStrings; QQmlEngine engine; }; typedef QList IntList; Q_DECLARE_METATYPE(IntList) typedef QList KeyList; Q_DECLARE_METATYPE(KeyList) Q_DECLARE_METATYPE(QQuickTextEdit::TextFormat) void tst_qquicktextedit::simulateKeys(QWindow *window, const QList &keys) { for (int i = 0; i < keys.count(); ++i) { const int key = keys.at(i).first; const int modifiers = key & Qt::KeyboardModifierMask; const QString text = !keys.at(i).second.isNull() ? QString(keys.at(i).second) : QString(); QKeyEvent press(QEvent::KeyPress, Qt::Key(key), Qt::KeyboardModifiers(modifiers), text); QKeyEvent release(QEvent::KeyRelease, Qt::Key(key), Qt::KeyboardModifiers(modifiers), text); QGuiApplication::sendEvent(window, &press); QGuiApplication::sendEvent(window, &release); } } void tst_qquicktextedit::simulateKeys(QWindow *window, const QKeySequence &sequence) { for (int i = 0; i < sequence.count(); ++i) { const int key = sequence[i]; const int modifiers = key & Qt::KeyboardModifierMask; QTest::keyClick(window, Qt::Key(key & ~modifiers), Qt::KeyboardModifiers(modifiers)); } } QList &operator <<(QList &keys, const QKeySequence &sequence) { for (int i = 0; i < sequence.count(); ++i) keys << Key(sequence[i], QChar()); return keys; } template QList &operator <<(QList &keys, const char (&characters)[N]) { for (int i = 0; i < N - 1; ++i) { int key = QTest::asciiToKey(characters[i]); QChar character = QLatin1Char(characters[i]); keys << Key(key, character); } return keys; } QList &operator <<(QList &keys, Qt::Key key) { keys << Key(key, QChar()); return keys; } tst_qquicktextedit::tst_qquicktextedit() { standard << "the quick brown fox jumped over the lazy dog" << "the quick brown fox\n jumped over the lazy dog" << "Hello, world!" << "!dlrow ,olleH"; richText << "the quick brown fox jumped over the lazy dog" << "the quick brown fox
jumped over the lazy dog
"; hAlignmentStrings << "AlignLeft" << "AlignRight" << "AlignHCenter"; vAlignmentStrings << "AlignTop" << "AlignBottom" << "AlignVCenter"; hAlignments << Qt::AlignLeft << Qt::AlignRight << Qt::AlignHCenter; vAlignments << Qt::AlignTop << Qt::AlignBottom << Qt::AlignVCenter; colorStrings << "aliceblue" << "antiquewhite" << "aqua" << "darkkhaki" << "darkolivegreen" << "dimgray" << "palevioletred" << "lightsteelblue" << "#000000" << "#AAAAAA" << "#FFFFFF" << "#2AC05F"; // // need a different test to do alpha channel test // << "#AA0011DD" // << "#00F16B11"; // } void tst_qquicktextedit::cleanup() { // ensure not even skipped tests with custom input context leave it dangling QInputMethodPrivate *inputMethodPrivate = QInputMethodPrivate::get(qApp->inputMethod()); inputMethodPrivate->testContext = 0; } void tst_qquicktextedit::text() { { QQmlComponent texteditComponent(&engine); texteditComponent.setData("import QtQuick 2.0\nTextEdit { text: \"\" }", QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QCOMPARE(textEditObject->text(), QString("")); QCOMPARE(textEditObject->length(), 0); } for (int i = 0; i < standard.size(); i++) { QString componentStr = "import QtQuick 2.0\nTextEdit { text: \"" + standard.at(i) + "\" }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QCOMPARE(textEditObject->text(), standard.at(i)); QCOMPARE(textEditObject->length(), standard.at(i).length()); } for (int i = 0; i < richText.size(); i++) { QString componentStr = "import QtQuick 2.0\nTextEdit { text: \"" + richText.at(i) + "\" }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QString expected = richText.at(i); expected.replace(QRegExp("\\\\(.)"),"\\1"); QCOMPARE(textEditObject->text(), expected); QCOMPARE(textEditObject->length(), expected.length()); } for (int i = 0; i < standard.size(); i++) { QString componentStr = "import QtQuick 2.0\nTextEdit { textFormat: TextEdit.RichText; text: \"" + standard.at(i) + "\" }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QString actual = textEditObject->text(); QString expected = standard.at(i); actual.remove(QRegExp(".*]*>")); actual.remove(QRegExp("(<[^>]*>)+")); expected.remove("\n"); QCOMPARE(actual.simplified(), expected); QCOMPARE(textEditObject->length(), expected.length()); } for (int i = 0; i < richText.size(); i++) { QString componentStr = "import QtQuick 2.0\nTextEdit { textFormat: TextEdit.RichText; text: \"" + richText.at(i) + "\" }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QString actual = textEditObject->text(); QString expected = richText.at(i); actual.replace(QRegExp(".*]*>"),""); actual.replace(QRegExp("(<[^>]*>)+"),"<>"); expected.replace(QRegExp("(<[^>]*>)+"),"<>"); QCOMPARE(actual.simplified(),expected.simplified()); expected.replace("<>", " "); QCOMPARE(textEditObject->length(), expected.simplified().length()); } for (int i = 0; i < standard.size(); i++) { QString componentStr = "import QtQuick 2.0\nTextEdit { textFormat: TextEdit.AutoText; text: \"" + standard.at(i) + "\" }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QCOMPARE(textEditObject->text(), standard.at(i)); QCOMPARE(textEditObject->length(), standard.at(i).length()); } for (int i = 0; i < richText.size(); i++) { QString componentStr = "import QtQuick 2.0\nTextEdit { textFormat: TextEdit.AutoText; text: \"" + richText.at(i) + "\" }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QString actual = textEditObject->text(); QString expected = richText.at(i); actual.replace(QRegExp(".*]*>"),""); actual.replace(QRegExp("(<[^>]*>)+"),"<>"); expected.replace(QRegExp("(<[^>]*>)+"),"<>"); QCOMPARE(actual.simplified(),expected.simplified()); expected.replace("<>", " "); QCOMPARE(textEditObject->length(), expected.simplified().length()); } } void tst_qquicktextedit::width() { // uses Font metrics to find the width for standard and document to find the width for rich { QQmlComponent texteditComponent(&engine); texteditComponent.setData("import QtQuick 2.0\nTextEdit { text: \"\" }", QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QCOMPARE(textEditObject->width(), 0.0); } bool requiresUnhintedMetrics = !qmlDisableDistanceField(); for (int i = 0; i < standard.size(); i++) { QString componentStr = "import QtQuick 2.0\nTextEdit { text: \"" + standard.at(i) + "\" }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QString s = standard.at(i); s.replace(QLatin1Char('\n'), QChar::LineSeparator); QTextLayout layout(s); layout.setFont(textEditObject->font()); layout.setFlags(Qt::TextExpandTabs | Qt::TextShowMnemonic); if (requiresUnhintedMetrics) { QTextOption option; option.setUseDesignMetrics(true); layout.setTextOption(option); } layout.beginLayout(); forever { QTextLine line = layout.createLine(); if (!line.isValid()) break; } layout.endLayout(); qreal metricWidth = layout.boundingRect().width(); QVERIFY(textEditObject != 0); QCOMPARE(textEditObject->width(), metricWidth); } for (int i = 0; i < richText.size(); i++) { QTextDocument document; document.setHtml(richText.at(i)); document.setDocumentMargin(0); if (requiresUnhintedMetrics) document.setUseDesignMetrics(true); qreal documentWidth = document.idealWidth(); QString componentStr = "import QtQuick 2.0\nTextEdit { textFormat: TextEdit.RichText; text: \"" + richText.at(i) + "\" }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QCOMPARE(textEditObject->width(), documentWidth); } } void tst_qquicktextedit::wrap() { // for specified width and wrap set true { QQmlComponent texteditComponent(&engine); texteditComponent.setData("import QtQuick 2.0\nTextEdit { text: \"\"; wrapMode: TextEdit.WordWrap; width: 300 }", QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QCOMPARE(textEditObject->width(), 300.); } for (int i = 0; i < standard.size(); i++) { QString componentStr = "import QtQuick 2.0\nTextEdit { wrapMode: TextEdit.WordWrap; width: 300; text: \"" + standard.at(i) + "\" }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QCOMPARE(textEditObject->width(), 300.); } for (int i = 0; i < richText.size(); i++) { QString componentStr = "import QtQuick 2.0\nTextEdit { wrapMode: TextEdit.WordWrap; width: 300; text: \"" + richText.at(i) + "\" }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QCOMPARE(textEditObject->width(), 300.); } } void tst_qquicktextedit::textFormat() { { QQmlComponent textComponent(&engine); textComponent.setData("import QtQuick 2.0\nTextEdit { text: \"Hello\"; textFormat: Text.RichText }", QUrl::fromLocalFile("")); QQuickTextEdit *textObject = qobject_cast(textComponent.create()); QVERIFY(textObject != 0); QVERIFY(textObject->textFormat() == QQuickTextEdit::RichText); } { QQmlComponent textComponent(&engine); textComponent.setData("import QtQuick 2.0\nTextEdit { text: \"Hello\"; textFormat: Text.PlainText }", QUrl::fromLocalFile("")); QQuickTextEdit *textObject = qobject_cast(textComponent.create()); QVERIFY(textObject != 0); QVERIFY(textObject->textFormat() == QQuickTextEdit::PlainText); } } void tst_qquicktextedit::alignments_data() { QTest::addColumn("hAlign"); QTest::addColumn("vAlign"); QTest::addColumn("expectfile"); QTest::newRow("LT") << int(Qt::AlignLeft) << int(Qt::AlignTop) << "alignments_lt"; QTest::newRow("RT") << int(Qt::AlignRight) << int(Qt::AlignTop) << "alignments_rt"; QTest::newRow("CT") << int(Qt::AlignHCenter) << int(Qt::AlignTop) << "alignments_ct"; QTest::newRow("LB") << int(Qt::AlignLeft) << int(Qt::AlignBottom) << "alignments_lb"; QTest::newRow("RB") << int(Qt::AlignRight) << int(Qt::AlignBottom) << "alignments_rb"; QTest::newRow("CB") << int(Qt::AlignHCenter) << int(Qt::AlignBottom) << "alignments_cb"; QTest::newRow("LC") << int(Qt::AlignLeft) << int(Qt::AlignVCenter) << "alignments_lc"; QTest::newRow("RC") << int(Qt::AlignRight) << int(Qt::AlignVCenter) << "alignments_rc"; QTest::newRow("CC") << int(Qt::AlignHCenter) << int(Qt::AlignVCenter) << "alignments_cc"; } void tst_qquicktextedit::alignments() { QSKIP("Image comparison of text is almost guaranteed to fail during development"); QFETCH(int, hAlign); QFETCH(int, vAlign); QFETCH(QString, expectfile); QQuickView canvas(testFileUrl("alignments.qml")); canvas.show(); canvas.requestActivateWindow(); QTest::qWaitForWindowShown(&canvas); QTRY_COMPARE(&canvas, qGuiApp->focusWindow()); QObject *ob = canvas.rootObject(); QVERIFY(ob != 0); ob->setProperty("horizontalAlignment",hAlign); ob->setProperty("verticalAlignment",vAlign); QTRY_COMPARE(ob->property("running").toBool(),false); QImage actual = canvas.grabFrameBuffer(); expectfile = createExpectedFileIfNotFound(expectfile, actual); QImage expect(expectfile); QCOMPARE(actual,expect); } //the alignment tests may be trivial o.oa void tst_qquicktextedit::hAlign() { //test one align each, and then test if two align fails. for (int i = 0; i < standard.size(); i++) { for (int j=0; j < hAlignmentStrings.size(); j++) { QString componentStr = "import QtQuick 2.0\nTextEdit { horizontalAlignment: \"" + hAlignmentStrings.at(j) + "\"; text: \"" + standard.at(i) + "\" }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QCOMPARE((int)textEditObject->hAlign(), (int)hAlignments.at(j)); } } for (int i = 0; i < richText.size(); i++) { for (int j=0; j < hAlignmentStrings.size(); j++) { QString componentStr = "import QtQuick 2.0\nTextEdit { horizontalAlignment: \"" + hAlignmentStrings.at(j) + "\"; text: \"" + richText.at(i) + "\" }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QCOMPARE((int)textEditObject->hAlign(), (int)hAlignments.at(j)); } } } void tst_qquicktextedit::hAlign_RightToLeft() { PlatformInputContext platformInputContext; QInputMethodPrivate *inputMethodPrivate = QInputMethodPrivate::get(qApp->inputMethod()); inputMethodPrivate->testContext = &platformInputContext; QQuickView canvas(testFileUrl("horizontalAlignment_RightToLeft.qml")); QQuickTextEdit *textEdit = canvas.rootObject()->findChild("text"); QVERIFY(textEdit != 0); canvas.show(); const QString rtlText = textEdit->text(); // implicit alignment should follow the reading direction of text QCOMPARE(textEdit->hAlign(), QQuickTextEdit::AlignRight); QVERIFY(textEdit->positionToRectangle(0).x() > canvas.width()/2); // explicitly left aligned textEdit->setHAlign(QQuickTextEdit::AlignLeft); QCOMPARE(textEdit->hAlign(), QQuickTextEdit::AlignLeft); QVERIFY(textEdit->positionToRectangle(0).x() < canvas.width()/2); // explicitly right aligned textEdit->setHAlign(QQuickTextEdit::AlignRight); QCOMPARE(textEdit->hAlign(), QQuickTextEdit::AlignRight); QVERIFY(textEdit->positionToRectangle(0).x() > canvas.width()/2); QString textString = textEdit->text(); textEdit->setText(QString("") + textString + QString("")); textEdit->resetHAlign(); // implicitly aligned rich text should follow the reading direction of RTL text QCOMPARE(textEdit->hAlign(), QQuickTextEdit::AlignRight); QCOMPARE(textEdit->effectiveHAlign(), textEdit->hAlign()); QVERIFY(textEdit->positionToRectangle(0).x() > canvas.width()/2); // explicitly left aligned rich text textEdit->setHAlign(QQuickTextEdit::AlignLeft); QCOMPARE(textEdit->hAlign(), QQuickTextEdit::AlignLeft); QCOMPARE(textEdit->effectiveHAlign(), textEdit->hAlign()); QVERIFY(textEdit->positionToRectangle(0).x() < canvas.width()/2); // explicitly right aligned rich text textEdit->setHAlign(QQuickTextEdit::AlignRight); QCOMPARE(textEdit->hAlign(), QQuickTextEdit::AlignRight); QCOMPARE(textEdit->effectiveHAlign(), textEdit->hAlign()); QVERIFY(textEdit->positionToRectangle(0).x() > canvas.width()/2); textEdit->setText(textString); // explicitly center aligned textEdit->setHAlign(QQuickTextEdit::AlignHCenter); QCOMPARE(textEdit->hAlign(), QQuickTextEdit::AlignHCenter); QVERIFY(textEdit->positionToRectangle(0).x() > canvas.width()/2); // reseted alignment should go back to following the text reading direction textEdit->resetHAlign(); QCOMPARE(textEdit->hAlign(), QQuickTextEdit::AlignRight); QVERIFY(textEdit->positionToRectangle(0).x() > canvas.width()/2); // mirror the text item QQuickItemPrivate::get(textEdit)->setLayoutMirror(true); // mirrored implicit alignment should continue to follow the reading direction of the text QCOMPARE(textEdit->hAlign(), QQuickTextEdit::AlignRight); QCOMPARE(textEdit->effectiveHAlign(), QQuickTextEdit::AlignRight); QVERIFY(textEdit->positionToRectangle(0).x() > canvas.width()/2); // mirrored explicitly right aligned behaves as left aligned textEdit->setHAlign(QQuickTextEdit::AlignRight); QCOMPARE(textEdit->hAlign(), QQuickTextEdit::AlignRight); QCOMPARE(textEdit->effectiveHAlign(), QQuickTextEdit::AlignLeft); QVERIFY(textEdit->positionToRectangle(0).x() < canvas.width()/2); // mirrored explicitly left aligned behaves as right aligned textEdit->setHAlign(QQuickTextEdit::AlignLeft); QCOMPARE(textEdit->hAlign(), QQuickTextEdit::AlignLeft); QCOMPARE(textEdit->effectiveHAlign(), QQuickTextEdit::AlignRight); QVERIFY(textEdit->positionToRectangle(0).x() > canvas.width()/2); // disable mirroring QQuickItemPrivate::get(textEdit)->setLayoutMirror(false); textEdit->resetHAlign(); // English text should be implicitly left aligned textEdit->setText("Hello world!"); QCOMPARE(textEdit->hAlign(), QQuickTextEdit::AlignLeft); QVERIFY(textEdit->positionToRectangle(0).x() < canvas.width()/2); canvas.requestActivateWindow(); QTest::qWaitForWindowShown(&canvas); QTRY_COMPARE(&canvas, qGuiApp->focusWindow()); textEdit->setText(QString()); { QInputMethodEvent ev(rtlText, QList()); QGuiApplication::sendEvent(qGuiApp->focusObject(), &ev); } QCOMPARE(textEdit->hAlign(), QQuickTextEdit::AlignRight); { QInputMethodEvent ev("Hello world!", QList()); QGuiApplication::sendEvent(qGuiApp->focusObject(), &ev); } QCOMPARE(textEdit->hAlign(), QQuickTextEdit::AlignLeft); // Clear pre-edit text. TextEdit should maybe do this itself on setText, but that may be // redundant as an actual input method may take care of it. { QInputMethodEvent ev; QGuiApplication::sendEvent(qGuiApp->focusObject(), &ev); } // empty text with implicit alignment follows the system locale-based // keyboard input direction from qApp->inputMethod()->inputDirection textEdit->setText(""); platformInputContext.setInputDirection(Qt::LeftToRight); QVERIFY(qApp->inputMethod()->inputDirection() == Qt::LeftToRight); QCOMPARE(textEdit->hAlign(), QQuickTextEdit::AlignLeft); QVERIFY(textEdit->positionToRectangle(0).x() < canvas.width()/2); QSignalSpy cursorRectangleSpy(textEdit, SIGNAL(cursorRectangleChanged())); platformInputContext.setInputDirection(Qt::RightToLeft); QCOMPARE(cursorRectangleSpy.count(), 1); QVERIFY(qApp->inputMethod()->inputDirection() == Qt::RightToLeft); QCOMPARE(textEdit->hAlign(), QQuickTextEdit::AlignRight); QVERIFY(textEdit->positionToRectangle(0).x() > canvas.width()/2); // set input direction while having content platformInputContext.setInputDirection(Qt::LeftToRight); textEdit->setText("a"); textEdit->setCursorPosition(1); platformInputContext.setInputDirection(Qt::RightToLeft); QTest::keyClick(&canvas, Qt::Key_Backspace); QVERIFY(textEdit->text().isEmpty()); QCOMPARE(textEdit->hAlign(), QQuickTextEdit::AlignRight); QVERIFY(textEdit->cursorRectangle().left() > canvas.width()/2); // input direction changed while not having focus platformInputContext.setInputDirection(Qt::LeftToRight); textEdit->setFocus(false); platformInputContext.setInputDirection(Qt::RightToLeft); textEdit->setFocus(true); QCOMPARE(textEdit->hAlign(), QQuickTextEdit::AlignRight); QVERIFY(textEdit->cursorRectangle().left() > canvas.width()/2); textEdit->setHAlign(QQuickTextEdit::AlignRight); QCOMPARE(textEdit->hAlign(), QQuickTextEdit::AlignRight); QVERIFY(textEdit->positionToRectangle(0).x() > canvas.width()/2); } void tst_qquicktextedit::vAlign() { //test one align each, and then test if two align fails. for (int i = 0; i < standard.size(); i++) { for (int j=0; j < vAlignmentStrings.size(); j++) { QString componentStr = "import QtQuick 2.0\nTextEdit { verticalAlignment: \"" + vAlignmentStrings.at(j) + "\"; text: \"" + standard.at(i) + "\" }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QCOMPARE((int)textEditObject->vAlign(), (int)vAlignments.at(j)); } } for (int i = 0; i < richText.size(); i++) { for (int j=0; j < vAlignmentStrings.size(); j++) { QString componentStr = "import QtQuick 2.0\nTextEdit { verticalAlignment: \"" + vAlignmentStrings.at(j) + "\"; text: \"" + richText.at(i) + "\" }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QCOMPARE((int)textEditObject->vAlign(), (int)vAlignments.at(j)); } } QQmlComponent texteditComponent(&engine); texteditComponent.setData( "import QtQuick 2.0\n" "TextEdit { width: 100; height: 100; text: \"Hello World\" }", QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QCOMPARE(textEditObject->vAlign(), QQuickTextEdit::AlignTop); QVERIFY(textEditObject->cursorRectangle().bottom() < 50); QVERIFY(textEditObject->positionToRectangle(0).bottom() < 50); // bottom aligned textEditObject->setVAlign(QQuickTextEdit::AlignBottom); QCOMPARE(textEditObject->vAlign(), QQuickTextEdit::AlignBottom); QVERIFY(textEditObject->cursorRectangle().top() > 50); QVERIFY(textEditObject->positionToRectangle(0).top() > 50); // explicitly center aligned textEditObject->setVAlign(QQuickTextEdit::AlignVCenter); QCOMPARE(textEditObject->vAlign(), QQuickTextEdit::AlignVCenter); QVERIFY(textEditObject->cursorRectangle().top() < 50); QVERIFY(textEditObject->cursorRectangle().bottom() > 50); QVERIFY(textEditObject->positionToRectangle(0).top() < 50); QVERIFY(textEditObject->positionToRectangle(0).bottom() > 50); } void tst_qquicktextedit::font() { //test size, then bold, then italic, then family { QString componentStr = "import QtQuick 2.0\nTextEdit { font.pointSize: 40; text: \"Hello World\" }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QCOMPARE(textEditObject->font().pointSize(), 40); QCOMPARE(textEditObject->font().bold(), false); QCOMPARE(textEditObject->font().italic(), false); } { QString componentStr = "import QtQuick 2.0\nTextEdit { font.bold: true; text: \"Hello World\" }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QCOMPARE(textEditObject->font().bold(), true); QCOMPARE(textEditObject->font().italic(), false); } { QString componentStr = "import QtQuick 2.0\nTextEdit { font.italic: true; text: \"Hello World\" }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QCOMPARE(textEditObject->font().italic(), true); QCOMPARE(textEditObject->font().bold(), false); } { QString componentStr = "import QtQuick 2.0\nTextEdit { font.family: \"Helvetica\"; text: \"Hello World\" }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QCOMPARE(textEditObject->font().family(), QString("Helvetica")); QCOMPARE(textEditObject->font().bold(), false); QCOMPARE(textEditObject->font().italic(), false); } { QString componentStr = "import QtQuick 2.0\nTextEdit { font.family: \"\"; text: \"Hello World\" }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QCOMPARE(textEditObject->font().family(), QString("")); } } void tst_qquicktextedit::color() { //test initial color { QString componentStr = "import QtQuick 2.0\nTextEdit { text: \"Hello World\" }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QQuickTextEditPrivate *textEditPrivate = static_cast(QQuickItemPrivate::get(textEditObject)); QVERIFY(textEditObject); QVERIFY(textEditPrivate); QVERIFY(textEditPrivate->control); QCOMPARE(textEditPrivate->color, QColor("black")); } //test normal for (int i = 0; i < colorStrings.size(); i++) { QString componentStr = "import QtQuick 2.0\nTextEdit { color: \"" + colorStrings.at(i) + "\"; text: \"Hello World\" }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); //qDebug() << "textEditObject: " << textEditObject->color() << "vs. " << QColor(colorStrings.at(i)); QVERIFY(textEditObject != 0); QCOMPARE(textEditObject->color(), QColor(colorStrings.at(i))); } //test selection for (int i = 0; i < colorStrings.size(); i++) { QString componentStr = "import QtQuick 2.0\nTextEdit { selectionColor: \"" + colorStrings.at(i) + "\"; text: \"Hello World\" }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QCOMPARE(textEditObject->selectionColor(), QColor(colorStrings.at(i))); } //test selected text for (int i = 0; i < colorStrings.size(); i++) { QString componentStr = "import QtQuick 2.0\nTextEdit { selectedTextColor: \"" + colorStrings.at(i) + "\"; text: \"Hello World\" }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QCOMPARE(textEditObject->selectedTextColor(), QColor(colorStrings.at(i))); } { QString colorStr = "#AA001234"; QColor testColor("#001234"); testColor.setAlpha(170); QString componentStr = "import QtQuick 2.0\nTextEdit { color: \"" + colorStr + "\"; text: \"Hello World\" }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QCOMPARE(textEditObject->color(), testColor); } } void tst_qquicktextedit::textMargin() { for (qreal i=0; i<=10; i+=0.3) { QString componentStr = "import QtQuick 2.0\nTextEdit { textMargin: " + QString::number(i) + "; text: \"Hello World\" }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QCOMPARE(textEditObject->textMargin(), i); } } void tst_qquicktextedit::persistentSelection() { QQuickView canvas(testFileUrl("persistentSelection.qml")); canvas.show(); canvas.requestActivateWindow(); QTest::qWaitForWindowShown(&canvas); QTRY_COMPARE(&canvas, qGuiApp->focusWindow()); canvas.requestActivateWindow(); QQuickTextEdit *edit = qobject_cast(canvas.rootObject()); QVERIFY(edit); QVERIFY(edit->hasActiveFocus()); QSignalSpy spy(edit, SIGNAL(persistentSelectionChanged(bool))); QCOMPARE(edit->persistentSelection(), false); edit->setPersistentSelection(false); QCOMPARE(edit->persistentSelection(), false); QCOMPARE(spy.count(), 0); edit->select(1, 4); QCOMPARE(edit->property("selected").toString(), QLatin1String("ell")); edit->setFocus(false); QCOMPARE(edit->property("selected").toString(), QString()); edit->setFocus(true); QCOMPARE(edit->property("selected").toString(), QString()); edit->setPersistentSelection(true); QCOMPARE(edit->persistentSelection(), true); QCOMPARE(spy.count(), 1); edit->select(1, 4); QCOMPARE(edit->property("selected").toString(), QLatin1String("ell")); edit->setFocus(false); QCOMPARE(edit->property("selected").toString(), QLatin1String("ell")); edit->setFocus(true); QCOMPARE(edit->property("selected").toString(), QLatin1String("ell")); } void tst_qquicktextedit::focusOnPress() { QString componentStr = "import QtQuick 2.0\n" "TextEdit {\n" "property bool selectOnFocus: false\n" "width: 100; height: 50\n" "activeFocusOnPress: true\n" "text: \"Hello World\"\n" "onFocusChanged: { if (focus && selectOnFocus) selectAll() }" " }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); QCOMPARE(textEditObject->focusOnPress(), true); QCOMPARE(textEditObject->hasFocus(), false); QSignalSpy focusSpy(textEditObject, SIGNAL(focusChanged(bool))); QSignalSpy activeFocusSpy(textEditObject, SIGNAL(focusChanged(bool))); QSignalSpy activeFocusOnPressSpy(textEditObject, SIGNAL(activeFocusOnPressChanged(bool))); textEditObject->setFocusOnPress(true); QCOMPARE(textEditObject->focusOnPress(), true); QCOMPARE(activeFocusOnPressSpy.count(), 0); QQuickCanvas canvas; canvas.resize(100, 50); textEditObject->setParentItem(canvas.rootItem()); canvas.show(); canvas.requestActivateWindow(); QTest::qWaitForWindowShown(&canvas); QTRY_COMPARE(QGuiApplication::activeWindow(), &canvas); QCOMPARE(textEditObject->hasFocus(), false); QCOMPARE(textEditObject->hasActiveFocus(), false); QPoint centerPoint(canvas.width()/2, canvas.height()/2); Qt::KeyboardModifiers noModifiers = 0; QTest::mousePress(&canvas, Qt::LeftButton, noModifiers, centerPoint); QGuiApplication::processEvents(); QCOMPARE(textEditObject->hasFocus(), true); QCOMPARE(textEditObject->hasActiveFocus(), true); QCOMPARE(focusSpy.count(), 1); QCOMPARE(activeFocusSpy.count(), 1); QCOMPARE(textEditObject->selectedText(), QString()); QTest::mouseRelease(&canvas, Qt::LeftButton, noModifiers, centerPoint); textEditObject->setFocusOnPress(false); QCOMPARE(textEditObject->focusOnPress(), false); QCOMPARE(activeFocusOnPressSpy.count(), 1); textEditObject->setFocus(false); QCOMPARE(textEditObject->hasFocus(), false); QCOMPARE(textEditObject->hasActiveFocus(), false); QCOMPARE(focusSpy.count(), 2); QCOMPARE(activeFocusSpy.count(), 2); // Wait for double click timeout to expire before clicking again. QTest::qWait(400); QTest::mousePress(&canvas, Qt::LeftButton, noModifiers, centerPoint); QGuiApplication::processEvents(); QCOMPARE(textEditObject->hasFocus(), false); QCOMPARE(textEditObject->hasActiveFocus(), false); QCOMPARE(focusSpy.count(), 2); QCOMPARE(activeFocusSpy.count(), 2); QTest::mouseRelease(&canvas, Qt::LeftButton, noModifiers, centerPoint); textEditObject->setFocusOnPress(true); QCOMPARE(textEditObject->focusOnPress(), true); QCOMPARE(activeFocusOnPressSpy.count(), 2); // Test a selection made in the on(Active)FocusChanged handler isn't overwritten. textEditObject->setProperty("selectOnFocus", true); QTest::qWait(400); QTest::mousePress(&canvas, Qt::LeftButton, noModifiers, centerPoint); QGuiApplication::processEvents(); QCOMPARE(textEditObject->hasFocus(), true); QCOMPARE(textEditObject->hasActiveFocus(), true); QCOMPARE(focusSpy.count(), 3); QCOMPARE(activeFocusSpy.count(), 3); QCOMPARE(textEditObject->selectedText(), textEditObject->text()); QTest::mouseRelease(&canvas, Qt::LeftButton, noModifiers, centerPoint); } void tst_qquicktextedit::selection() { QString testStr = standard[0];//TODO: What should happen for multiline/rich text? QString componentStr = "import QtQuick 2.0\nTextEdit { text: \""+ testStr +"\"; }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEditObject = qobject_cast(texteditComponent.create()); QVERIFY(textEditObject != 0); //Test selection follows cursor for (int i=0; i<= testStr.size(); i++) { textEditObject->setCursorPosition(i); QCOMPARE(textEditObject->cursorPosition(), i); QCOMPARE(textEditObject->selectionStart(), i); QCOMPARE(textEditObject->selectionEnd(), i); QVERIFY(textEditObject->selectedText().isNull()); } textEditObject->setCursorPosition(0); QVERIFY(textEditObject->cursorPosition() == 0); QVERIFY(textEditObject->selectionStart() == 0); QVERIFY(textEditObject->selectionEnd() == 0); QVERIFY(textEditObject->selectedText().isNull()); // Verify invalid positions are ignored. textEditObject->setCursorPosition(-1); QVERIFY(textEditObject->cursorPosition() == 0); QVERIFY(textEditObject->selectionStart() == 0); QVERIFY(textEditObject->selectionEnd() == 0); QVERIFY(textEditObject->selectedText().isNull()); textEditObject->setCursorPosition(textEditObject->text().count()+1); QVERIFY(textEditObject->cursorPosition() == 0); QVERIFY(textEditObject->selectionStart() == 0); QVERIFY(textEditObject->selectionEnd() == 0); QVERIFY(textEditObject->selectedText().isNull()); //Test selection for (int i=0; i<= testStr.size(); i++) { textEditObject->select(0,i); QCOMPARE(testStr.mid(0,i), textEditObject->selectedText()); } for (int i=0; i<= testStr.size(); i++) { textEditObject->select(i,testStr.size()); QCOMPARE(testStr.mid(i,testStr.size()-i), textEditObject->selectedText()); } textEditObject->setCursorPosition(0); QVERIFY(textEditObject->cursorPosition() == 0); QVERIFY(textEditObject->selectionStart() == 0); QVERIFY(textEditObject->selectionEnd() == 0); QVERIFY(textEditObject->selectedText().isNull()); //Test Error Ignoring behaviour textEditObject->setCursorPosition(0); QVERIFY(textEditObject->selectedText().isNull()); textEditObject->select(-10,0); QVERIFY(textEditObject->selectedText().isNull()); textEditObject->select(100,101); QVERIFY(textEditObject->selectedText().isNull()); textEditObject->select(0,-10); QVERIFY(textEditObject->selectedText().isNull()); textEditObject->select(0,100); QVERIFY(textEditObject->selectedText().isNull()); textEditObject->select(0,10); QVERIFY(textEditObject->selectedText().size() == 10); textEditObject->select(-10,0); QVERIFY(textEditObject->selectedText().size() == 10); textEditObject->select(100,101); QVERIFY(textEditObject->selectedText().size() == 10); textEditObject->select(0,-10); QVERIFY(textEditObject->selectedText().size() == 10); textEditObject->select(0,100); QVERIFY(textEditObject->selectedText().size() == 10); textEditObject->deselect(); QVERIFY(textEditObject->selectedText().isNull()); textEditObject->select(0,10); QVERIFY(textEditObject->selectedText().size() == 10); textEditObject->deselect(); QVERIFY(textEditObject->selectedText().isNull()); } void tst_qquicktextedit::isRightToLeft_data() { QTest::addColumn("text"); QTest::addColumn("emptyString"); QTest::addColumn("firstCharacter"); QTest::addColumn("lastCharacter"); QTest::addColumn("middleCharacter"); QTest::addColumn("startString"); QTest::addColumn("midString"); QTest::addColumn("endString"); const quint16 arabic_str[] = { 0x0638, 0x0643, 0x00646, 0x0647, 0x0633, 0x0638, 0x0643, 0x00646, 0x0647, 0x0633, 0x0647}; QTest::newRow("Empty") << "" << false << false << false << false << false << false << false; QTest::newRow("Neutral") << "23244242" << false << false << false << false << false << false << false; QTest::newRow("LTR") << "Hello world" << false << false << false << false << false << false << false; QTest::newRow("RTL") << QString::fromUtf16(arabic_str, 11) << false << true << true << true << true << true << true; QTest::newRow("Bidi RTL + LTR + RTL") << QString::fromUtf16(arabic_str, 11) + QString("Hello world") + QString::fromUtf16(arabic_str, 11) << false << true << true << false << true << true << true; QTest::newRow("Bidi LTR + RTL + LTR") << QString("Hello world") + QString::fromUtf16(arabic_str, 11) + QString("Hello world") << false << false << false << true << false << false << false; } void tst_qquicktextedit::isRightToLeft() { QFETCH(QString, text); QFETCH(bool, emptyString); QFETCH(bool, firstCharacter); QFETCH(bool, lastCharacter); QFETCH(bool, middleCharacter); QFETCH(bool, startString); QFETCH(bool, midString); QFETCH(bool, endString); QQuickTextEdit textEdit; textEdit.setText(text); // first test that the right string is delivered to the QString::isRightToLeft() QCOMPARE(textEdit.isRightToLeft(0,0), text.mid(0,0).isRightToLeft()); QCOMPARE(textEdit.isRightToLeft(0,1), text.mid(0,1).isRightToLeft()); QCOMPARE(textEdit.isRightToLeft(text.count()-2, text.count()-1), text.mid(text.count()-2, text.count()-1).isRightToLeft()); QCOMPARE(textEdit.isRightToLeft(text.count()/2, text.count()/2 + 1), text.mid(text.count()/2, text.count()/2 + 1).isRightToLeft()); QCOMPARE(textEdit.isRightToLeft(0,text.count()/4), text.mid(0,text.count()/4).isRightToLeft()); QCOMPARE(textEdit.isRightToLeft(text.count()/4,3*text.count()/4), text.mid(text.count()/4,3*text.count()/4).isRightToLeft()); if (text.isEmpty()) QTest::ignoreMessage(QtWarningMsg, ": QML TextEdit: isRightToLeft(start, end) called with the end property being smaller than the start."); QCOMPARE(textEdit.isRightToLeft(3*text.count()/4,text.count()-1), text.mid(3*text.count()/4,text.count()-1).isRightToLeft()); // then test that the feature actually works QCOMPARE(textEdit.isRightToLeft(0,0), emptyString); QCOMPARE(textEdit.isRightToLeft(0,1), firstCharacter); QCOMPARE(textEdit.isRightToLeft(text.count()-2, text.count()-1), lastCharacter); QCOMPARE(textEdit.isRightToLeft(text.count()/2, text.count()/2 + 1), middleCharacter); QCOMPARE(textEdit.isRightToLeft(0,text.count()/4), startString); QCOMPARE(textEdit.isRightToLeft(text.count()/4,3*text.count()/4), midString); if (text.isEmpty()) QTest::ignoreMessage(QtWarningMsg, ": QML TextEdit: isRightToLeft(start, end) called with the end property being smaller than the start."); QCOMPARE(textEdit.isRightToLeft(3*text.count()/4,text.count()-1), endString); } void tst_qquicktextedit::keySelection() { QQuickView canvas(testFileUrl("navigation.qml")); canvas.show(); canvas.requestActivateWindow(); QTest::qWaitForWindowShown(&canvas); QTRY_COMPARE(&canvas, qGuiApp->focusWindow()); canvas.requestActivateWindow(); QVERIFY(canvas.rootObject() != 0); QQuickTextEdit *input = qobject_cast(qvariant_cast(canvas.rootObject()->property("myInput"))); QVERIFY(input != 0); QTRY_VERIFY(input->hasActiveFocus() == true); QSignalSpy spy(input, SIGNAL(selectionChanged())); simulateKey(&canvas, Qt::Key_Right, Qt::ShiftModifier); QVERIFY(input->hasActiveFocus() == true); QCOMPARE(input->selectedText(), QString("a")); QCOMPARE(spy.count(), 1); simulateKey(&canvas, Qt::Key_Right); QVERIFY(input->hasActiveFocus() == true); QCOMPARE(input->selectedText(), QString()); QCOMPARE(spy.count(), 2); simulateKey(&canvas, Qt::Key_Right); QVERIFY(input->hasActiveFocus() == false); QCOMPARE(input->selectedText(), QString()); QCOMPARE(spy.count(), 2); simulateKey(&canvas, Qt::Key_Left); QVERIFY(input->hasActiveFocus() == true); QCOMPARE(spy.count(), 2); simulateKey(&canvas, Qt::Key_Left, Qt::ShiftModifier); QVERIFY(input->hasActiveFocus() == true); QCOMPARE(input->selectedText(), QString("a")); QCOMPARE(spy.count(), 3); simulateKey(&canvas, Qt::Key_Left); QVERIFY(input->hasActiveFocus() == true); QCOMPARE(input->selectedText(), QString()); QCOMPARE(spy.count(), 4); simulateKey(&canvas, Qt::Key_Left); QVERIFY(input->hasActiveFocus() == false); QCOMPARE(input->selectedText(), QString()); QCOMPARE(spy.count(), 4); } void tst_qquicktextedit::moveCursorSelection_data() { QTest::addColumn("testStr"); QTest::addColumn("cursorPosition"); QTest::addColumn("movePosition"); QTest::addColumn("mode"); QTest::addColumn("selectionStart"); QTest::addColumn("selectionEnd"); QTest::addColumn("reversible"); QTest::newRow("(t)he|characters") << standard[0] << 0 << 1 << QQuickTextEdit::SelectCharacters << 0 << 1 << true; QTest::newRow("do(g)|characters") << standard[0] << 43 << 44 << QQuickTextEdit::SelectCharacters << 43 << 44 << true; QTest::newRow("jum(p)ed|characters") << standard[0] << 23 << 24 << QQuickTextEdit::SelectCharacters << 23 << 24 << true; QTest::newRow("jumped( )over|characters") << standard[0] << 26 << 27 << QQuickTextEdit::SelectCharacters << 26 << 27 << true; QTest::newRow("(the )|characters") << standard[0] << 0 << 4 << QQuickTextEdit::SelectCharacters << 0 << 4 << true; QTest::newRow("( dog)|characters") << standard[0] << 40 << 44 << QQuickTextEdit::SelectCharacters << 40 << 44 << true; QTest::newRow("( jumped )|characters") << standard[0] << 19 << 27 << QQuickTextEdit::SelectCharacters << 19 << 27 << true; QTest::newRow("th(e qu)ick|characters") << standard[0] << 2 << 6 << QQuickTextEdit::SelectCharacters << 2 << 6 << true; QTest::newRow("la(zy d)og|characters") << standard[0] << 38 << 42 << QQuickTextEdit::SelectCharacters << 38 << 42 << true; QTest::newRow("jum(ped ov)er|characters") << standard[0] << 23 << 29 << QQuickTextEdit::SelectCharacters << 23 << 29 << true; QTest::newRow("()the|characters") << standard[0] << 0 << 0 << QQuickTextEdit::SelectCharacters << 0 << 0 << true; QTest::newRow("dog()|characters") << standard[0] << 44 << 44 << QQuickTextEdit::SelectCharacters << 44 << 44 << true; QTest::newRow("jum()ped|characters") << standard[0] << 23 << 23 << QQuickTextEdit::SelectCharacters << 23 << 23 << true; QTest::newRow("<(t)he>|words") << standard[0] << 0 << 1 << QQuickTextEdit::SelectWords << 0 << 3 << true; QTest::newRow("|words") << standard[0] << 43 << 44 << QQuickTextEdit::SelectWords << 41 << 44 << true; QTest::newRow("|words") << standard[0] << 23 << 24 << QQuickTextEdit::SelectWords << 20 << 26 << true; QTest::newRow("over|words") << standard[0] << 26 << 27 << QQuickTextEdit::SelectWords << 20 << 27 << false; QTest::newRow("jumped<( )over>|words,reversed") << standard[0] << 27 << 26 << QQuickTextEdit::SelectWords << 26 << 31 << false; QTest::newRow("<(the )>quick|words") << standard[0] << 0 << 4 << QQuickTextEdit::SelectWords << 0 << 4 << false; QTest::newRow("<(the )quick>|words,reversed") << standard[0] << 4 << 0 << QQuickTextEdit::SelectWords << 0 << 9 << false; QTest::newRow("|words") << standard[0] << 40 << 44 << QQuickTextEdit::SelectWords << 36 << 44 << false; QTest::newRow("lazy<( dog)>|words,reversed") << standard[0] << 44 << 40 << QQuickTextEdit::SelectWords << 40 << 44 << false; QTest::newRow("over|words") << standard[0] << 19 << 27 << QQuickTextEdit::SelectWords << 16 << 27 << false; QTest::newRow("fox<( jumped )over>|words,reversed") << standard[0] << 27 << 19 << QQuickTextEdit::SelectWords << 19 << 31 << false; QTest::newRow("|words") << standard[0] << 2 << 6 << QQuickTextEdit::SelectWords << 0 << 9 << true; QTest::newRow("") << standard[0] << 38 << 42 << QQuickTextEdit::SelectWords << 36 << 44 << true; QTest::newRow("|words") << standard[0] << 23 << 29 << QQuickTextEdit::SelectWords << 20 << 31 << true; QTest::newRow("<()>the|words") << standard[0] << 0 << 0 << QQuickTextEdit::SelectWords << 0 << 0 << true; QTest::newRow("dog<()>|words") << standard[0] << 44 << 44 << QQuickTextEdit::SelectWords << 44 << 44 << true; QTest::newRow("jum<()>ped|words") << standard[0] << 23 << 23 << QQuickTextEdit::SelectWords << 23 << 23 << true; QTest::newRow("Hello<(,)> |words") << standard[2] << 5 << 6 << QQuickTextEdit::SelectWords << 5 << 6 << true; QTest::newRow("Hello<(, )>world|words") << standard[2] << 5 << 7 << QQuickTextEdit::SelectWords << 5 << 7 << false; QTest::newRow("Hello<(, )world>|words,reversed") << standard[2] << 7 << 5 << QQuickTextEdit::SelectWords << 5 << 12 << false; QTest::newRow("world|words") << standard[2] << 3 << 7 << QQuickTextEdit::SelectWords << 0 << 7 << false; QTest::newRow("|words,reversed") << standard[2] << 7 << 3 << QQuickTextEdit::SelectWords << 0 << 12 << false; QTest::newRow(",|words") << standard[2] << 3 << 5 << QQuickTextEdit::SelectWords << 0 << 5 << true; QTest::newRow("Hello<()>,|words") << standard[2] << 5 << 5 << QQuickTextEdit::SelectWords << 5 << 5 << true; QTest::newRow("Hello,<()>|words") << standard[2] << 6 << 6 << QQuickTextEdit::SelectWords << 6 << 6 << true; QTest::newRow("Hello<,( )>world|words") << standard[2] << 6 << 7 << QQuickTextEdit::SelectWords << 5 << 7 << false; QTest::newRow("Hello,<( )world>|words,reversed") << standard[2] << 7 << 6 << QQuickTextEdit::SelectWords << 6 << 12 << false; QTest::newRow("Hello<,( world)>|words") << standard[2] << 6 << 12 << QQuickTextEdit::SelectWords << 5 << 12 << false; QTest::newRow("Hello,<( world)>|words,reversed") << standard[2] << 12 << 6 << QQuickTextEdit::SelectWords << 6 << 12 << false; QTest::newRow("Hello<,( world!)>|words") << standard[2] << 6 << 13 << QQuickTextEdit::SelectWords << 5 << 13 << false; QTest::newRow("Hello,<( world!)>|words,reversed") << standard[2] << 13 << 6 << QQuickTextEdit::SelectWords << 6 << 13 << false; QTest::newRow("Hello<(, world!)>|words") << standard[2] << 5 << 13 << QQuickTextEdit::SelectWords << 5 << 13 << true; QTest::newRow("world<(!)>|words") << standard[2] << 12 << 13 << QQuickTextEdit::SelectWords << 12 << 13 << true; QTest::newRow("world!<()>)|words") << standard[2] << 13 << 13 << QQuickTextEdit::SelectWords << 13 << 13 << true; QTest::newRow("world<()>!)|words") << standard[2] << 12 << 12 << QQuickTextEdit::SelectWords << 12 << 12 << true; QTest::newRow("<(,)>olleH |words") << standard[3] << 7 << 8 << QQuickTextEdit::SelectWords << 7 << 8 << true; QTest::newRow("olleH|words") << standard[3] << 6 << 8 << QQuickTextEdit::SelectWords << 1 << 8 << false; QTest::newRow("dlrow<( ,)>olleH|words,reversed") << standard[3] << 8 << 6 << QQuickTextEdit::SelectWords << 6 << 8 << false; QTest::newRow("|words") << standard[3] << 6 << 10 << QQuickTextEdit::SelectWords << 1 << 13 << false; QTest::newRow("dlrow<( ,ol)leH>|words,reversed") << standard[3] << 10 << 6 << QQuickTextEdit::SelectWords << 6 << 13 << false; QTest::newRow(",<(ol)leH>,|words") << standard[3] << 8 << 10 << QQuickTextEdit::SelectWords << 8 << 13 << true; QTest::newRow(",<()>olleH|words") << standard[3] << 8 << 8 << QQuickTextEdit::SelectWords << 8 << 8 << true; QTest::newRow("<()>,olleH|words") << standard[3] << 7 << 7 << QQuickTextEdit::SelectWords << 7 << 7 << true; QTest::newRow(",olleH|words") << standard[3] << 6 << 7 << QQuickTextEdit::SelectWords << 1 << 7 << false; QTest::newRow("dlrow<( ),>olleH|words,reversed") << standard[3] << 7 << 6 << QQuickTextEdit::SelectWords << 6 << 8 << false; QTest::newRow("<(dlrow )>,olleH|words") << standard[3] << 1 << 7 << QQuickTextEdit::SelectWords << 1 << 7 << false; QTest::newRow("<(dlrow ),>olleH|words,reversed") << standard[3] << 7 << 1 << QQuickTextEdit::SelectWords << 1 << 8 << false; QTest::newRow("<(!dlrow )>,olleH|words") << standard[3] << 0 << 7 << QQuickTextEdit::SelectWords << 0 << 7 << false; QTest::newRow("<(!dlrow ),>olleH|words,reversed") << standard[3] << 7 << 0 << QQuickTextEdit::SelectWords << 0 << 8 << false; QTest::newRow("(!dlrow ,)olleH|words") << standard[3] << 0 << 8 << QQuickTextEdit::SelectWords << 0 << 8 << true; QTest::newRow("<(!)>dlrow|words") << standard[3] << 0 << 1 << QQuickTextEdit::SelectWords << 0 << 1 << true; QTest::newRow("<()>!dlrow|words") << standard[3] << 0 << 0 << QQuickTextEdit::SelectWords << 0 << 0 << true; QTest::newRow("!<()>dlrow|words") << standard[3] << 1 << 1 << QQuickTextEdit::SelectWords << 1 << 1 << true; } void tst_qquicktextedit::moveCursorSelection() { QFETCH(QString, testStr); QFETCH(int, cursorPosition); QFETCH(int, movePosition); QFETCH(QQuickTextEdit::SelectionMode, mode); QFETCH(int, selectionStart); QFETCH(int, selectionEnd); QFETCH(bool, reversible); QString componentStr = "import QtQuick 2.0\nTextEdit { text: \""+ testStr +"\"; }"; QQmlComponent textinputComponent(&engine); textinputComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *texteditObject = qobject_cast(textinputComponent.create()); QVERIFY(texteditObject != 0); texteditObject->setCursorPosition(cursorPosition); texteditObject->moveCursorSelection(movePosition, mode); QCOMPARE(texteditObject->selectedText(), testStr.mid(selectionStart, selectionEnd - selectionStart)); QCOMPARE(texteditObject->selectionStart(), selectionStart); QCOMPARE(texteditObject->selectionEnd(), selectionEnd); if (reversible) { texteditObject->setCursorPosition(movePosition); texteditObject->moveCursorSelection(cursorPosition, mode); QCOMPARE(texteditObject->selectedText(), testStr.mid(selectionStart, selectionEnd - selectionStart)); QCOMPARE(texteditObject->selectionStart(), selectionStart); QCOMPARE(texteditObject->selectionEnd(), selectionEnd); } } void tst_qquicktextedit::moveCursorSelectionSequence_data() { QTest::addColumn("testStr"); QTest::addColumn("cursorPosition"); QTest::addColumn("movePosition1"); QTest::addColumn("movePosition2"); QTest::addColumn("selection1Start"); QTest::addColumn("selection1End"); QTest::addColumn("selection2Start"); QTest::addColumn("selection2End"); QTest::newRow("the { f^ox} jumped|ltr") << standard[0] << 9 << 13 << 17 << 4 << 15 << 4 << 19; QTest::newRow("the quick<( {bro)wn> f^ox} jumped|rtl") << standard[0] << 13 << 9 << 17 << 9 << 15 << 10 << 19; QTest::newRow("the { ^}fox jumped|ltr") << standard[0] << 9 << 13 << 16 << 4 << 15 << 4 << 16; QTest::newRow("the quick<( {bro)wn> ^}fox jumped|rtl") << standard[0] << 13 << 9 << 16 << 9 << 15 << 10 << 16; QTest::newRow("the {} fox jumped|ltr") << standard[0] << 9 << 13 << 15 << 4 << 15 << 4 << 15; QTest::newRow("the quick<( {bro)wn^>} f^ox jumped|rtl") << standard[0] << 13 << 9 << 15 << 9 << 15 << 10 << 15; QTest::newRow("the { fox|ltr") << standard[0] << 9 << 13 << 10 << 4 << 15 << 4 << 10; QTest::newRow("the quick<(^ {^bro)wn>} fox|rtl") << standard[0] << 13 << 9 << 10 << 9 << 15 << 10 << 15; QTest::newRow("the { fox|ltr") << standard[0] << 9 << 13 << 9 << 4 << 15 << 4 << 9; QTest::newRow("the quick{<(^ bro)wn>} fox|rtl") << standard[0] << 13 << 9 << 9 << 9 << 15 << 9 << 15; QTest::newRow("the { fox|ltr") << standard[0] << 9 << 13 << 7 << 4 << 15 << 4 << 9; QTest::newRow("the { fox|rtl") << standard[0] << 13 << 9 << 7 << 9 << 15 << 4 << 15; QTest::newRow("the {<^quick}( bro)wn> fox|ltr") << standard[0] << 9 << 13 << 4 << 4 << 15 << 4 << 9; QTest::newRow("the {<^quick}( bro)wn> fox|rtl") << standard[0] << 13 << 9 << 4 << 9 << 15 << 4 << 15; QTest::newRow("the{^ fox|ltr") << standard[0] << 9 << 13 << 3 << 4 << 15 << 3 << 9; QTest::newRow("the{^ fox|rtl") << standard[0] << 13 << 9 << 3 << 9 << 15 << 3 << 15; QTest::newRow("{t^he fox|ltr") << standard[0] << 9 << 13 << 1 << 4 << 15 << 0 << 9; QTest::newRow("{t^he fox|rtl") << standard[0] << 13 << 9 << 1 << 9 << 15 << 0 << 15; QTest::newRow("{, w^orld}!|ltr") << standard[2] << 2 << 4 << 8 << 0 << 5 << 0 << 12; QTest::newRow("{, w^orld}!|rtl") << standard[2] << 4 << 2 << 8 << 0 << 5 << 0 << 12; QTest::newRow("!{dlro^w ,}|ltr") << standard[3] << 9 << 11 << 5 << 8 << 13 << 1 << 13; QTest::newRow("!{dlro^w ,}|rtl") << standard[3] << 11 << 9 << 5 << 8 << 13 << 1 << 13; } void tst_qquicktextedit::moveCursorSelectionSequence() { QFETCH(QString, testStr); QFETCH(int, cursorPosition); QFETCH(int, movePosition1); QFETCH(int, movePosition2); QFETCH(int, selection1Start); QFETCH(int, selection1End); QFETCH(int, selection2Start); QFETCH(int, selection2End); QString componentStr = "import QtQuick 2.0\nTextEdit { text: \""+ testStr +"\"; }"; QQmlComponent texteditComponent(&engine); texteditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *texteditObject = qobject_cast(texteditComponent.create()); QVERIFY(texteditObject != 0); texteditObject->setCursorPosition(cursorPosition); texteditObject->moveCursorSelection(movePosition1, QQuickTextEdit::SelectWords); QCOMPARE(texteditObject->selectedText(), testStr.mid(selection1Start, selection1End - selection1Start)); QCOMPARE(texteditObject->selectionStart(), selection1Start); QCOMPARE(texteditObject->selectionEnd(), selection1End); texteditObject->moveCursorSelection(movePosition2, QQuickTextEdit::SelectWords); QCOMPARE(texteditObject->selectedText(), testStr.mid(selection2Start, selection2End - selection2Start)); QCOMPARE(texteditObject->selectionStart(), selection2Start); QCOMPARE(texteditObject->selectionEnd(), selection2End); } void tst_qquicktextedit::mouseSelection_data() { QTest::addColumn("qmlfile"); QTest::addColumn("from"); QTest::addColumn("to"); QTest::addColumn("selectedText"); QTest::addColumn("focus"); QTest::addColumn("focusOnPress"); QTest::addColumn("doubleClick"); // import installed QTest::newRow("on") << testFile("mouseselection_true.qml") << 4 << 9 << "45678" << true << true << false; QTest::newRow("off") << testFile("mouseselection_false.qml") << 4 << 9 << QString() << true << true << false; QTest::newRow("default") << testFile("mouseselection_default.qml") << 4 << 9 << QString() << true << true << false; QTest::newRow("off word selection") << testFile("mouseselection_false_words.qml") << 4 << 9 << QString() << true << true << false; QTest::newRow("on word selection (4,9)") << testFile("mouseselection_true_words.qml") << 4 << 9 << "0123456789" << true << true << false; QTest::newRow("on unfocused") << testFile("mouseselection_true.qml") << 4 << 9 << "45678" << false << false << false; QTest::newRow("on word selection (4,9) unfocused") << testFile("mouseselection_true_words.qml") << 4 << 9 << "0123456789" << false << false << false; QTest::newRow("on focus on press") << testFile("mouseselection_true.qml") << 4 << 9 << "45678" << false << true << false; QTest::newRow("on word selection (4,9) focus on press") << testFile("mouseselection_true_words.qml") << 4 << 9 << "0123456789" << false << true << false; QTest::newRow("on word selection (2,13)") << testFile("mouseselection_true_words.qml") << 2 << 13 << "0123456789 ABCDEFGHIJKLMNOPQRSTUVWXYZ" << true << true << false; QTest::newRow("on word selection (2,30)") << testFile("mouseselection_true_words.qml") << 2 << 30 << "0123456789 ABCDEFGHIJKLMNOPQRSTUVWXYZ" << true << true << false; QTest::newRow("on word selection (9,13)") << testFile("mouseselection_true_words.qml") << 9 << 13 << "0123456789 ABCDEFGHIJKLMNOPQRSTUVWXYZ" << true << true << false; QTest::newRow("on word selection (9,30)") << testFile("mouseselection_true_words.qml") << 9 << 30 << "0123456789 ABCDEFGHIJKLMNOPQRSTUVWXYZ" << true << true << false; QTest::newRow("on word selection (13,2)") << testFile("mouseselection_true_words.qml") << 13 << 2 << "0123456789 ABCDEFGHIJKLMNOPQRSTUVWXYZ" << true << true << false; QTest::newRow("on word selection (20,2)") << testFile("mouseselection_true_words.qml") << 20 << 2 << "0123456789 ABCDEFGHIJKLMNOPQRSTUVWXYZ" << true << true << false; QTest::newRow("on word selection (12,9)") << testFile("mouseselection_true_words.qml") << 12 << 9 << "0123456789 ABCDEFGHIJKLMNOPQRSTUVWXYZ" << true << true << false; QTest::newRow("on word selection (30,9)") << testFile("mouseselection_true_words.qml") << 30 << 9 << "0123456789 ABCDEFGHIJKLMNOPQRSTUVWXYZ" << true << true << false; QTest::newRow("off double click (4,9)") << testFile("mouseselection_true.qml") << 4 << 9 << "0123456789" << true << true << true; QTest::newRow("off double click (2,13)") << testFile("mouseselection_true.qml") << 2 << 13 << "0123456789 ABCDEFGHIJKLMNOPQRSTUVWXYZ" << true << true << true; QTest::newRow("off double click (2,30)") << testFile("mouseselection_true.qml") << 2 << 30 << "0123456789 ABCDEFGHIJKLMNOPQRSTUVWXYZ" << true << true << true; QTest::newRow("off double click (9,13)") << testFile("mouseselection_true.qml") << 9 << 13 << "0123456789 ABCDEFGHIJKLMNOPQRSTUVWXYZ" << true << true << true; QTest::newRow("off double click (9,30)") << testFile("mouseselection_true.qml") << 9 << 30 << "0123456789 ABCDEFGHIJKLMNOPQRSTUVWXYZ" << true << true << true; QTest::newRow("off double click (13,2)") << testFile("mouseselection_true.qml") << 13 << 2 << "0123456789 ABCDEFGHIJKLMNOPQRSTUVWXYZ" << true << true << true; QTest::newRow("off double click (20,2)") << testFile("mouseselection_true.qml") << 20 << 2 << "0123456789 ABCDEFGHIJKLMNOPQRSTUVWXYZ" << true << true << true; QTest::newRow("off double click (12,9)") << testFile("mouseselection_true.qml") << 12 << 9 << "0123456789 ABCDEFGHIJKLMNOPQRSTUVWXYZ" << true << true << true; QTest::newRow("off double click (30,9)") << testFile("mouseselection_true.qml") << 30 << 9 << "0123456789 ABCDEFGHIJKLMNOPQRSTUVWXYZ" << true << true << true; } void tst_qquicktextedit::mouseSelection() { QFETCH(QString, qmlfile); QFETCH(int, from); QFETCH(int, to); QFETCH(QString, selectedText); QFETCH(bool, focus); QFETCH(bool, focusOnPress); QFETCH(bool, doubleClick); QQuickView canvas(QUrl::fromLocalFile(qmlfile)); canvas.show(); canvas.requestActivateWindow(); QTest::qWaitForWindowShown(&canvas); QTRY_COMPARE(&canvas, qGuiApp->focusWindow()); QVERIFY(canvas.rootObject() != 0); QQuickTextEdit *textEditObject = qobject_cast(canvas.rootObject()); QVERIFY(textEditObject != 0); textEditObject->setFocus(focus); textEditObject->setFocusOnPress(focusOnPress); // press-and-drag-and-release from x1 to x2 QPoint p1 = textEditObject->positionToRectangle(from).center().toPoint(); QPoint p2 = textEditObject->positionToRectangle(to).center().toPoint(); if (doubleClick) QTest::mouseClick(&canvas, Qt::LeftButton, 0, p1); QTest::mousePress(&canvas, Qt::LeftButton, 0, p1); QTest::mouseMove(&canvas, p2); QTest::mouseRelease(&canvas, Qt::LeftButton, 0, p2); QTest::qWait(50); QTRY_COMPARE(textEditObject->selectedText(), selectedText); // Clicking and shift to clicking between the same points should select the same text. textEditObject->setCursorPosition(0); if (doubleClick) QTest::mouseDClick(&canvas, Qt::LeftButton, 0, p1); else QTest::mouseClick(&canvas, Qt::LeftButton, Qt::NoModifier, p1); QTest::mouseClick(&canvas, Qt::LeftButton, Qt::ShiftModifier, p2); QTest::qWait(50); QTRY_COMPARE(textEditObject->selectedText(), selectedText); } void tst_qquicktextedit::dragMouseSelection() { QString qmlfile = testFile("mouseselection_true.qml"); QQuickView canvas(QUrl::fromLocalFile(qmlfile)); canvas.show(); canvas.requestActivateWindow(); QTest::qWaitForWindowShown(&canvas); QTRY_COMPARE(&canvas, qGuiApp->focusWindow()); QVERIFY(canvas.rootObject() != 0); QQuickTextEdit *textEditObject = qobject_cast(canvas.rootObject()); QVERIFY(textEditObject != 0); // press-and-drag-and-release from x1 to x2 int x1 = 10; int x2 = 70; int y = textEditObject->height()/2; QTest::mousePress(&canvas, Qt::LeftButton, 0, QPoint(x1,y)); QTest::mouseMove(&canvas, QPoint(x2, y)); QTest::mouseRelease(&canvas, Qt::LeftButton, 0, QPoint(x2,y)); QTest::qWait(300); QString str1; QTRY_VERIFY((str1 = textEditObject->selectedText()).length() > 3); // press and drag the current selection. x1 = 40; x2 = 100; QTest::mousePress(&canvas, Qt::LeftButton, 0, QPoint(x1,y)); QTest::mouseMove(&canvas, QPoint(x2, y)); QTest::mouseRelease(&canvas, Qt::LeftButton, 0, QPoint(x2,y)); QTest::qWait(300); QString str2; QTRY_VERIFY((str2 = textEditObject->selectedText()).length() > 3); QVERIFY(str1 != str2); // Verify the second press and drag is a new selection and not the first moved. } void tst_qquicktextedit::mouseSelectionMode_data() { QTest::addColumn("qmlfile"); QTest::addColumn("selectWords"); // import installed QTest::newRow("SelectWords") << testFile("mouseselectionmode_words.qml") << true; QTest::newRow("SelectCharacters") << testFile("mouseselectionmode_characters.qml") << false; QTest::newRow("default") << testFile("mouseselectionmode_default.qml") << false; } void tst_qquicktextedit::mouseSelectionMode() { QFETCH(QString, qmlfile); QFETCH(bool, selectWords); QString text = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ"; QQuickView canvas(QUrl::fromLocalFile(qmlfile)); canvas.show(); canvas.requestActivateWindow(); QTest::qWaitForWindowShown(&canvas); QTRY_COMPARE(&canvas, qGuiApp->focusWindow()); QVERIFY(canvas.rootObject() != 0); QQuickTextEdit *textEditObject = qobject_cast(canvas.rootObject()); QVERIFY(textEditObject != 0); // press-and-drag-and-release from x1 to x2 int x1 = 10; int x2 = 70; int y = textEditObject->height()/2; QTest::mousePress(&canvas, Qt::LeftButton, 0, QPoint(x1,y)); QTest::mouseMove(&canvas, QPoint(x2, y)); //QTest::mouseMove(canvas, QPoint(x2,y)); // doesn't work // QMouseEvent mv(QEvent::MouseMove, QPoint(x2,y), Qt::LeftButton, Qt::LeftButton,Qt::NoModifier); // QGuiApplication::sendEvent(&canvas, &mv); QTest::mouseRelease(&canvas, Qt::LeftButton, 0, QPoint(x2,y)); QString str = textEditObject->selectedText(); if (selectWords) { QTRY_COMPARE(textEditObject->selectedText(), text); } else { QTRY_VERIFY(textEditObject->selectedText().length() > 3); QVERIFY(str != text); } } void tst_qquicktextedit::inputMethodHints() { QQuickView canvas(testFileUrl("inputmethodhints.qml")); canvas.show(); canvas.requestActivateWindow(); QVERIFY(canvas.rootObject() != 0); QQuickTextEdit *textEditObject = qobject_cast(canvas.rootObject()); QVERIFY(textEditObject != 0); QVERIFY(textEditObject->inputMethodHints() & Qt::ImhNoPredictiveText); QSignalSpy inputMethodHintSpy(textEditObject, SIGNAL(inputMethodHintsChanged())); textEditObject->setInputMethodHints(Qt::ImhUppercaseOnly); QVERIFY(textEditObject->inputMethodHints() & Qt::ImhUppercaseOnly); QCOMPARE(inputMethodHintSpy.count(), 1); textEditObject->setInputMethodHints(Qt::ImhUppercaseOnly); QCOMPARE(inputMethodHintSpy.count(), 1); QQuickTextEdit plainTextEdit; QCOMPARE(plainTextEdit.inputMethodHints(), Qt::ImhNone); } void tst_qquicktextedit::positionAt() { QQuickView canvas(testFileUrl("positionAt.qml")); QVERIFY(canvas.rootObject() != 0); canvas.show(); canvas.requestActivateWindow(); canvas.requestActivateWindow(); QTest::qWaitForWindowShown(&canvas); QQuickTextEdit *texteditObject = qobject_cast(canvas.rootObject()); QVERIFY(texteditObject != 0); QTextLayout layout(texteditObject->text()); layout.setFont(texteditObject->font()); if (!qmlDisableDistanceField()) { QTextOption option; option.setUseDesignMetrics(true); layout.setTextOption(option); } layout.beginLayout(); QTextLine line = layout.createLine(); layout.endLayout(); const int y0 = line.height() / 2; const int y1 = line.height() * 3 / 2; int pos = texteditObject->positionAt(texteditObject->width()/2, y0); int widthBegin = floor(line.cursorToX(pos - 1)); int widthEnd = ceil(line.cursorToX(pos + 1)); QVERIFY(widthBegin <= texteditObject->width() / 2); QVERIFY(widthEnd >= texteditObject->width() / 2); const qreal x0 = texteditObject->positionToRectangle(pos).x(); const qreal x1 = texteditObject->positionToRectangle(pos + 1).x(); QString preeditText = texteditObject->text().mid(0, pos); texteditObject->setText(texteditObject->text().mid(pos)); texteditObject->setCursorPosition(0); QInputMethodEvent inputEvent(preeditText, QList()); QGuiApplication::sendEvent(qGuiApp->focusObject(), &inputEvent); // Check all points within the preedit text return the same position. QCOMPARE(texteditObject->positionAt(0, y0), 0); QCOMPARE(texteditObject->positionAt(x0 / 2, y0), 0); QCOMPARE(texteditObject->positionAt(x0, y0), 0); // Verify positioning returns to normal after the preedit text. QCOMPARE(texteditObject->positionAt(x1, y0), 1); QCOMPARE(texteditObject->positionToRectangle(1).x(), x1); QVERIFY(texteditObject->positionAt(x0 / 2, y1) > 0); } void tst_qquicktextedit::linkActivated() { QQuickView canvas(testFileUrl("linkActivated.qml")); QVERIFY(canvas.rootObject() != 0); canvas.show(); canvas.requestActivateWindow(); QTest::qWaitForWindowShown(&canvas); QQuickTextEdit *texteditObject = qobject_cast(canvas.rootObject()); QVERIFY(texteditObject != 0); QSignalSpy spy(texteditObject, SIGNAL(linkActivated(QString))); const QString link("http://example.com/"); const QPointF linkPos = texteditObject->positionToRectangle(7).center(); const QPointF textPos = texteditObject->positionToRectangle(2).center(); QTest::mouseClick(&canvas, Qt::LeftButton, 0, linkPos.toPoint()); QTRY_COMPARE(spy.count(), 1); QCOMPARE(spy.last()[0].toString(), link); QTest::mouseClick(&canvas, Qt::LeftButton, 0, textPos.toPoint()); QTest::qWait(50); QCOMPARE(spy.count(), 1); texteditObject->setReadOnly(true); QTest::mouseClick(&canvas, Qt::LeftButton, 0, linkPos.toPoint()); QTRY_COMPARE(spy.count(), 2); QCOMPARE(spy.last()[0].toString(), link); QTest::mouseClick(&canvas, Qt::LeftButton, 0, textPos.toPoint()); QTest::qWait(50); QCOMPARE(spy.count(), 2); } void tst_qquicktextedit::cursorDelegate_data() { QTest::addColumn("source"); QTest::newRow("out of line") << testFileUrl("cursorTest.qml"); QTest::newRow("in line") << testFileUrl("cursorTestInline.qml"); QTest::newRow("external") << testFileUrl("cursorTestExternal.qml"); } void tst_qquicktextedit::cursorDelegate() { QFETCH(QUrl, source); QQuickView view(source); view.show(); view.requestActivateWindow(); QQuickTextEdit *textEditObject = view.rootObject()->findChild("textEditObject"); QVERIFY(textEditObject != 0); QVERIFY(textEditObject->findChild("cursorInstance")); //Test Delegate gets created textEditObject->setFocus(true); QQuickItem* delegateObject = textEditObject->findChild("cursorInstance"); QVERIFY(delegateObject); QCOMPARE(delegateObject->property("localProperty").toString(), QString("Hello")); //Test Delegate gets moved for (int i=0; i<= textEditObject->text().length(); i++) { textEditObject->setCursorPosition(i); QCOMPARE(textEditObject->cursorRectangle().x(), delegateObject->x()); QCOMPARE(textEditObject->cursorRectangle().y(), delegateObject->y()); } // Clear preedit text; QInputMethodEvent event; QGuiApplication::sendEvent(&view, &event); // Test delegate gets moved on mouse press. textEditObject->setSelectByMouse(true); textEditObject->setCursorPosition(0); const QPoint point1 = textEditObject->positionToRectangle(5).center().toPoint(); QTest::qWait(400); //ensure this isn't treated as a double-click QTest::mouseClick(&view, Qt::LeftButton, 0, point1); QTest::qWait(50); QTRY_VERIFY(textEditObject->cursorPosition() != 0); QCOMPARE(textEditObject->cursorRectangle().x(), delegateObject->x()); QCOMPARE(textEditObject->cursorRectangle().y(), delegateObject->y()); // Test delegate gets moved on mouse drag textEditObject->setCursorPosition(0); const QPoint point2 = textEditObject->positionToRectangle(10).center().toPoint(); QTest::qWait(400); //ensure this isn't treated as a double-click QTest::mousePress(&view, Qt::LeftButton, 0, point1); QMouseEvent mv(QEvent::MouseMove, point2, Qt::LeftButton, Qt::LeftButton,Qt::NoModifier); QGuiApplication::sendEvent(&view, &mv); QTest::mouseRelease(&view, Qt::LeftButton, 0, point2); QTest::qWait(50); QTRY_COMPARE(textEditObject->cursorRectangle().x(), delegateObject->x()); QCOMPARE(textEditObject->cursorRectangle().y(), delegateObject->y()); textEditObject->setReadOnly(true); textEditObject->setCursorPosition(0); QTest::qWait(400); //ensure this isn't treated as a double-click QTest::mouseClick(&view, Qt::LeftButton, 0, textEditObject->positionToRectangle(5).center().toPoint()); QTest::qWait(50); QTRY_VERIFY(textEditObject->cursorPosition() != 0); QCOMPARE(textEditObject->cursorRectangle().x(), delegateObject->x()); QCOMPARE(textEditObject->cursorRectangle().y(), delegateObject->y()); textEditObject->setCursorPosition(0); QTest::qWait(400); //ensure this isn't treated as a double-click QTest::mouseClick(&view, Qt::LeftButton, 0, textEditObject->positionToRectangle(5).center().toPoint()); QTest::qWait(50); QTRY_VERIFY(textEditObject->cursorPosition() != 0); QCOMPARE(textEditObject->cursorRectangle().x(), delegateObject->x()); QCOMPARE(textEditObject->cursorRectangle().y(), delegateObject->y()); textEditObject->setCursorPosition(0); QCOMPARE(textEditObject->cursorRectangle().x(), delegateObject->x()); QCOMPARE(textEditObject->cursorRectangle().y(), delegateObject->y()); //Test Delegate gets deleted textEditObject->setCursorDelegate(0); QVERIFY(!textEditObject->findChild("cursorInstance")); } void tst_qquicktextedit::cursorVisible() { QQuickView view(testFileUrl("cursorVisible.qml")); view.show(); view.requestActivateWindow(); QTest::qWaitForWindowShown(&view); QTRY_COMPARE(&view, qGuiApp->focusWindow()); QQuickTextEdit edit; QSignalSpy spy(&edit, SIGNAL(cursorVisibleChanged(bool))); QCOMPARE(edit.isCursorVisible(), false); edit.setCursorVisible(true); QCOMPARE(edit.isCursorVisible(), true); QCOMPARE(spy.count(), 1); edit.setCursorVisible(false); QCOMPARE(edit.isCursorVisible(), false); QCOMPARE(spy.count(), 2); edit.setFocus(true); QCOMPARE(edit.isCursorVisible(), false); QCOMPARE(spy.count(), 2); edit.setParentItem(view.rootObject()); QCOMPARE(edit.isCursorVisible(), true); QCOMPARE(spy.count(), 3); edit.setFocus(false); QCOMPARE(edit.isCursorVisible(), false); QCOMPARE(spy.count(), 4); edit.setFocus(true); QCOMPARE(edit.isCursorVisible(), true); QCOMPARE(spy.count(), 5); QQuickView alternateView; alternateView.show(); alternateView.requestActivateWindow(); QTest::qWaitForWindowShown(&alternateView); QCOMPARE(edit.isCursorVisible(), false); QCOMPARE(spy.count(), 6); view.requestActivateWindow(); QTest::qWaitForWindowShown(&view); QCOMPARE(edit.isCursorVisible(), true); QCOMPARE(spy.count(), 7); } void tst_qquicktextedit::delegateLoading_data() { QTest::addColumn("qmlfile"); QTest::addColumn("error"); // import installed QTest::newRow("pass") << "cursorHttpTestPass.qml" << ""; QTest::newRow("fail1") << "cursorHttpTestFail1.qml" << "http://localhost:42332/FailItem.qml: Remote host closed the connection "; QTest::newRow("fail2") << "cursorHttpTestFail2.qml" << "http://localhost:42332/ErrItem.qml:4:5: Fungus is not a type "; } void tst_qquicktextedit::delegateLoading() { #ifdef Q_OS_MAC QSKIP("Test crashes during canvas tear down. QTBUG-23010"); #endif QFETCH(QString, qmlfile); QFETCH(QString, error); TestHTTPServer server(42332); server.serveDirectory(testFile("httpfail"), TestHTTPServer::Disconnect); server.serveDirectory(testFile("httpslow"), TestHTTPServer::Delay); server.serveDirectory(testFile("http")); QQuickView view(QUrl(QLatin1String("http://localhost:42332/") + qmlfile)); view.show(); view.requestActivateWindow(); if (!error.isEmpty()) { QTest::ignoreMessage(QtWarningMsg, error.toUtf8()); QTRY_VERIFY(view.status()==QQuickView::Error); QTRY_VERIFY(!view.rootObject()); // there is fail item inside this test } else { QTRY_VERIFY(view.rootObject());//Wait for loading to finish. QQuickTextEdit *textEditObject = view.rootObject()->findChild("textEditObject"); // view.rootObject()->dumpObjectTree(); QVERIFY(textEditObject != 0); textEditObject->setFocus(true); QQuickItem *delegate; delegate = view.rootObject()->findChild("delegateOkay"); QVERIFY(delegate); delegate = view.rootObject()->findChild("delegateSlow"); QVERIFY(delegate); delete delegate; } //A test should be added here with a component which is ready but component.create() returns null //Not sure how to accomplish this with QQuickTextEdits cursor delegate //###This was only needed for code coverage, and could be a case of overzealous defensive programming //delegate = view.rootObject()->findChild("delegateErrorB"); //QVERIFY(!delegate); } /* TextEdit element should only handle left/right keys until the cursor reaches the extent of the text, then they should ignore the keys. */ void tst_qquicktextedit::navigation() { QQuickView canvas(testFileUrl("navigation.qml")); canvas.show(); canvas.requestActivateWindow(); QVERIFY(canvas.rootObject() != 0); QQuickItem *input = qobject_cast(qvariant_cast(canvas.rootObject()->property("myInput"))); QVERIFY(input != 0); QTRY_VERIFY(input->hasActiveFocus() == true); simulateKey(&canvas, Qt::Key_Left); QVERIFY(input->hasActiveFocus() == false); simulateKey(&canvas, Qt::Key_Right); QVERIFY(input->hasActiveFocus() == true); simulateKey(&canvas, Qt::Key_Right); QVERIFY(input->hasActiveFocus() == true); simulateKey(&canvas, Qt::Key_Right); QVERIFY(input->hasActiveFocus() == false); simulateKey(&canvas, Qt::Key_Left); QVERIFY(input->hasActiveFocus() == true); } void tst_qquicktextedit::copyAndPaste() { #ifndef QT_NO_CLIPBOARD #ifdef Q_OS_MAC { PasteboardRef pasteboard; OSStatus status = PasteboardCreate(0, &pasteboard); if (status == noErr) CFRelease(pasteboard); else QSKIP("This machine doesn't support the clipboard"); } #endif QString componentStr = "import QtQuick 2.0\nTextEdit { text: \"Hello world!\" }"; QQmlComponent textEditComponent(&engine); textEditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEdit = qobject_cast(textEditComponent.create()); QVERIFY(textEdit != 0); // copy and paste QCOMPARE(textEdit->text().length(), 12); textEdit->select(0, textEdit->text().length());; textEdit->copy(); QCOMPARE(textEdit->selectedText(), QString("Hello world!")); QCOMPARE(textEdit->selectedText().length(), 12); textEdit->setCursorPosition(0); QVERIFY(textEdit->canPaste()); textEdit->paste(); QCOMPARE(textEdit->text(), QString("Hello world!Hello world!")); QCOMPARE(textEdit->text().length(), 24); // canPaste QVERIFY(textEdit->canPaste()); textEdit->setReadOnly(true); QVERIFY(!textEdit->canPaste()); textEdit->setReadOnly(false); QVERIFY(textEdit->canPaste()); // QTBUG-12339 // test that document and internal text attribute are in sync QQuickItemPrivate* pri = QQuickItemPrivate::get(textEdit); QQuickTextEditPrivate *editPrivate = static_cast(pri); QCOMPARE(textEdit->text(), editPrivate->text); // select word textEdit->setCursorPosition(0); textEdit->selectWord(); QCOMPARE(textEdit->selectedText(), QString("Hello")); // select all and cut textEdit->selectAll(); textEdit->cut(); QCOMPARE(textEdit->text().length(), 0); textEdit->paste(); QCOMPARE(textEdit->text(), QString("Hello world!Hello world!")); QCOMPARE(textEdit->text().length(), 24); #endif } void tst_qquicktextedit::canPaste() { #ifndef QT_NO_CLIPBOARD QGuiApplication::clipboard()->setText("Some text"); QString componentStr = "import QtQuick 2.0\nTextEdit { text: \"Hello world!\" }"; QQmlComponent textEditComponent(&engine); textEditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEdit = qobject_cast(textEditComponent.create()); QVERIFY(textEdit != 0); // check initial value - QTBUG-17765 QQuickTextControl tc(0); QCOMPARE(textEdit->canPaste(), tc.canPaste()); #endif } void tst_qquicktextedit::canPasteEmpty() { #ifndef QT_NO_CLIPBOARD QGuiApplication::clipboard()->clear(); QString componentStr = "import QtQuick 2.0\nTextEdit { text: \"Hello world!\" }"; QQmlComponent textEditComponent(&engine); textEditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEdit = qobject_cast(textEditComponent.create()); QVERIFY(textEdit != 0); // check initial value - QTBUG-17765 QQuickTextControl tc(0); QCOMPARE(textEdit->canPaste(), tc.canPaste()); #endif } void tst_qquicktextedit::readOnly() { QQuickView canvas(testFileUrl("readOnly.qml")); canvas.show(); canvas.requestActivateWindow(); QVERIFY(canvas.rootObject() != 0); QQuickTextEdit *edit = qobject_cast(qvariant_cast(canvas.rootObject()->property("myInput"))); QVERIFY(edit != 0); QTRY_VERIFY(edit->hasActiveFocus() == true); QVERIFY(edit->isReadOnly() == true); QString initial = edit->text(); for (int k=Qt::Key_0; k<=Qt::Key_Z; k++) simulateKey(&canvas, k); simulateKey(&canvas, Qt::Key_Return); simulateKey(&canvas, Qt::Key_Space); simulateKey(&canvas, Qt::Key_Escape); QCOMPARE(edit->text(), initial); edit->setCursorPosition(3); edit->setReadOnly(false); QCOMPARE(edit->isReadOnly(), false); QCOMPARE(edit->cursorPosition(), edit->text().length()); } void tst_qquicktextedit::simulateKey(QWindow *view, int key, Qt::KeyboardModifiers modifiers) { QKeyEvent press(QKeyEvent::KeyPress, key, modifiers); QKeyEvent release(QKeyEvent::KeyRelease, key, modifiers); QGuiApplication::sendEvent(view, &press); QGuiApplication::sendEvent(view, &release); } void tst_qquicktextedit::textInput() { QQuickView view(testFileUrl("inputMethodEvent.qml")); view.show(); view.requestActivateWindow(); QTest::qWaitForWindowShown(&view); QTRY_COMPARE(&view, qGuiApp->focusWindow()); QQuickTextEdit *edit = qobject_cast(view.rootObject()); QVERIFY(edit); QVERIFY(edit->hasActiveFocus() == true); // test that input method event is committed and change signal is emitted QSignalSpy spy(edit, SIGNAL(textChanged())); QInputMethodEvent event; event.setCommitString( "Hello world!", 0, 0); QGuiApplication::sendEvent(qGuiApp->focusObject(), &event); QCOMPARE(edit->text(), QString("Hello world!")); QCOMPARE(spy.count(), 1); // QTBUG-12339 // test that document and internal text attribute are in sync QQuickTextEditPrivate *editPrivate = static_cast(QQuickItemPrivate::get(edit)); QCOMPARE(editPrivate->text, QString("Hello world!")); // test that tentative commit is included in text property edit->setText(""); spy.clear(); QList attributes; QInputMethodEvent event2("preedit", attributes); event2.setTentativeCommitString("string"); QGuiApplication::sendEvent(qGuiApp->focusObject(), &event2); QCOMPARE(spy.count(), 1); QCOMPARE(edit->text(), QString("string")); QInputMethodQueryEvent queryEvent(Qt::ImEnabled); QGuiApplication::sendEvent(qGuiApp->focusObject(), &queryEvent); QCOMPARE(queryEvent.value(Qt::ImEnabled).toBool(), true); edit->setReadOnly(true); QGuiApplication::sendEvent(edit, &queryEvent); QCOMPARE(queryEvent.value(Qt::ImEnabled).toBool(), false); } void tst_qquicktextedit::inputMethodUpdate() { PlatformInputContext platformInputContext; QInputMethodPrivate *inputMethodPrivate = QInputMethodPrivate::get(qApp->inputMethod()); inputMethodPrivate->testContext = &platformInputContext; QQuickView view(testFileUrl("inputMethodEvent.qml")); view.show(); view.requestActivateWindow(); QTest::qWaitForWindowShown(&view); QTRY_COMPARE(&view, qGuiApp->focusWindow()); QQuickTextEdit *edit = qobject_cast(view.rootObject()); QVERIFY(edit); QVERIFY(edit->hasActiveFocus() == true); // text change even without cursor position change needs to trigger update edit->setText("test"); platformInputContext.clear(); edit->setText("xxxx"); QVERIFY(platformInputContext.m_updateCallCount > 0); // input method event replacing text platformInputContext.clear(); { QInputMethodEvent inputMethodEvent; inputMethodEvent.setCommitString("y", -1, 1); QGuiApplication::sendEvent(edit, &inputMethodEvent); } QVERIFY(platformInputContext.m_updateCallCount > 0); // input method changing selection platformInputContext.clear(); { QList attributes; attributes << QInputMethodEvent::Attribute(QInputMethodEvent::Selection, 0, 2, QVariant()); QInputMethodEvent inputMethodEvent("", attributes); QGuiApplication::sendEvent(edit, &inputMethodEvent); } QVERIFY(edit->selectionStart() != edit->selectionEnd()); QVERIFY(platformInputContext.m_updateCallCount > 0); // font changes platformInputContext.clear(); QFont font = edit->font(); font.setBold(!font.bold()); edit->setFont(font); QVERIFY(platformInputContext.m_updateCallCount > 0); // normal input platformInputContext.clear(); { QInputMethodEvent inputMethodEvent; inputMethodEvent.setCommitString("y"); QGuiApplication::sendEvent(edit, &inputMethodEvent); } QVERIFY(platformInputContext.m_updateCallCount > 0); // changing cursor position platformInputContext.clear(); edit->setCursorPosition(0); QVERIFY(platformInputContext.m_updateCallCount > 0); // continuing with selection platformInputContext.clear(); edit->moveCursorSelection(1); QVERIFY(platformInputContext.m_updateCallCount > 0); // read only disabled input method platformInputContext.clear(); edit->setReadOnly(true); QVERIFY(platformInputContext.m_updateCallCount > 0); edit->setReadOnly(false); // no updates while no focus edit->setFocus(false); platformInputContext.clear(); edit->setText("Foo"); QCOMPARE(platformInputContext.m_updateCallCount, 0); edit->setCursorPosition(1); QCOMPARE(platformInputContext.m_updateCallCount, 0); edit->selectAll(); QCOMPARE(platformInputContext.m_updateCallCount, 0); edit->setReadOnly(true); QCOMPARE(platformInputContext.m_updateCallCount, 0); } void tst_qquicktextedit::openInputPanel() { PlatformInputContext platformInputContext; QInputMethodPrivate *inputMethodPrivate = QInputMethodPrivate::get(qApp->inputMethod()); inputMethodPrivate->testContext = &platformInputContext; QQuickView view(testFileUrl("openInputPanel.qml")); view.show(); view.requestActivateWindow(); QTest::qWaitForWindowShown(&view); QTRY_COMPARE(&view, qGuiApp->focusWindow()); QQuickTextEdit *edit = qobject_cast(view.rootObject()); QVERIFY(edit); // check default values QVERIFY(edit->focusOnPress()); QVERIFY(!edit->hasActiveFocus()); qDebug() << &edit << qApp->focusObject(); QVERIFY(qApp->focusObject() != edit); QCOMPARE(qApp->inputMethod()->visible(), false); // input panel should open on focus QPoint centerPoint(view.width()/2, view.height()/2); Qt::KeyboardModifiers noModifiers = 0; QTest::mousePress(&view, Qt::LeftButton, noModifiers, centerPoint); QGuiApplication::processEvents(); QVERIFY(edit->hasActiveFocus()); QCOMPARE(qApp->focusObject(), edit); QCOMPARE(qApp->inputMethod()->visible(), true); QTest::mouseRelease(&view, Qt::LeftButton, noModifiers, centerPoint); // input panel should be re-opened when pressing already focused TextEdit qApp->inputMethod()->hide(); QCOMPARE(qApp->inputMethod()->visible(), false); QVERIFY(edit->hasActiveFocus()); QTest::mousePress(&view, Qt::LeftButton, noModifiers, centerPoint); QGuiApplication::processEvents(); QCOMPARE(qApp->inputMethod()->visible(), true); QTest::mouseRelease(&view, Qt::LeftButton, noModifiers, centerPoint); // input panel should stay visible if focus is lost to another text editor QSignalSpy inputPanelVisibilitySpy(qApp->inputMethod(), SIGNAL(visibleChanged())); QQuickTextEdit anotherEdit; anotherEdit.setParentItem(view.rootObject()); anotherEdit.setFocus(true); QCOMPARE(qApp->inputMethod()->visible(), true); QCOMPARE(qApp->focusObject(), qobject_cast(&anotherEdit)); QCOMPARE(inputPanelVisibilitySpy.count(), 0); anotherEdit.setFocus(false); QVERIFY(qApp->focusObject() != &anotherEdit); QCOMPARE(view.activeFocusItem(), view.rootItem()); anotherEdit.setFocus(true); qApp->inputMethod()->hide(); // input panel should not be opened if TextEdit is read only edit->setReadOnly(true); edit->setFocus(true); QCOMPARE(qApp->inputMethod()->visible(), false); QTest::mousePress(&view, Qt::LeftButton, noModifiers, centerPoint); QTest::mouseRelease(&view, Qt::LeftButton, noModifiers, centerPoint); QGuiApplication::processEvents(); QCOMPARE(qApp->inputMethod()->visible(), false); // input panel should not be opened if focusOnPress is set to false edit->setFocusOnPress(false); edit->setFocus(false); edit->setFocus(true); QCOMPARE(qApp->inputMethod()->visible(), false); QTest::mousePress(&view, Qt::LeftButton, noModifiers, centerPoint); QTest::mouseRelease(&view, Qt::LeftButton, noModifiers, centerPoint); QCOMPARE(qApp->inputMethod()->visible(), false); // input panel should open when openSoftwareInputPanel is called edit->openSoftwareInputPanel(); QCOMPARE(qApp->inputMethod()->visible(), true); // input panel should close when closeSoftwareInputPanel is called edit->closeSoftwareInputPanel(); QCOMPARE(qApp->inputMethod()->visible(), false); inputMethodPrivate->testContext = 0; } void tst_qquicktextedit::geometrySignals() { QQmlComponent component(&engine, testFileUrl("geometrySignals.qml")); QObject *o = component.create(); QVERIFY(o); QCOMPARE(o->property("bindingWidth").toInt(), 400); QCOMPARE(o->property("bindingHeight").toInt(), 500); delete o; } void tst_qquicktextedit::pastingRichText_QTBUG_14003() { #ifndef QT_NO_CLIPBOARD QString componentStr = "import QtQuick 2.0\nTextEdit { textFormat: TextEdit.PlainText }"; QQmlComponent component(&engine); component.setData(componentStr.toLatin1(), QUrl::fromLocalFile("")); QQuickTextEdit *obj = qobject_cast(component.create()); QTRY_VERIFY(obj != 0); QTRY_VERIFY(obj->textFormat() == QQuickTextEdit::PlainText); QMimeData *mData = new QMimeData; mData->setHtml("Hello"); QGuiApplication::clipboard()->setMimeData(mData); obj->paste(); QTRY_VERIFY(obj->text() == ""); QTRY_VERIFY(obj->textFormat() == QQuickTextEdit::PlainText); #endif } void tst_qquicktextedit::implicitSize_data() { QTest::addColumn("text"); QTest::addColumn("wrap"); QTest::addColumn("format"); QTest::newRow("plain") << "The quick red fox jumped over the lazy brown dog" << "TextEdit.NoWrap" << "TextEdit.PlainText"; QTest::newRow("richtext") << "The quick red fox jumped over the lazy brown dog" << "TextEdit.NoWrap" << "TextEdit.RichText"; QTest::newRow("plain_wrap") << "The quick red fox jumped over the lazy brown dog" << "TextEdit.Wrap" << "TextEdit.PlainText"; QTest::newRow("richtext_wrap") << "The quick red fox jumped over the lazy brown dog" << "TextEdit.Wrap" << "TextEdit.RichText"; } void tst_qquicktextedit::implicitSize() { QFETCH(QString, text); QFETCH(QString, wrap); QFETCH(QString, format); QString componentStr = "import QtQuick 2.0\nTextEdit { text: \"" + text + "\"; width: 50; wrapMode: " + wrap + "; textFormat: " + format + " }"; QQmlComponent textComponent(&engine); textComponent.setData(componentStr.toLatin1(), QUrl::fromLocalFile("")); QQuickTextEdit *textObject = qobject_cast(textComponent.create()); QVERIFY(textObject->width() < textObject->implicitWidth()); QVERIFY(textObject->height() == textObject->implicitHeight()); textObject->resetWidth(); QVERIFY(textObject->width() == textObject->implicitWidth()); QVERIFY(textObject->height() == textObject->implicitHeight()); } void tst_qquicktextedit::contentSize() { QString componentStr = "import QtQuick 2.0\nTextEdit { width: 75; height: 16; font.pixelSize: 10 }"; QQmlComponent textComponent(&engine); textComponent.setData(componentStr.toLatin1(), QUrl::fromLocalFile("")); QScopedPointer object(textComponent.create()); QQuickTextEdit *textObject = qobject_cast(object.data()); QSignalSpy spy(textObject, SIGNAL(contentSizeChanged())); textObject->setText("The quick red fox jumped over the lazy brown dog"); QVERIFY(textObject->contentWidth() > textObject->width()); QVERIFY(textObject->contentHeight() < textObject->height()); QCOMPARE(spy.count(), 1); textObject->setWrapMode(QQuickTextEdit::WordWrap); QVERIFY(textObject->contentWidth() <= textObject->width()); QVERIFY(textObject->contentHeight() > textObject->height()); QCOMPARE(spy.count(), 2); textObject->setText("The quickredfoxjumpedoverthe lazy brown dog"); QVERIFY(textObject->contentWidth() > textObject->width()); QVERIFY(textObject->contentHeight() > textObject->height()); QCOMPARE(spy.count(), 3); } void tst_qquicktextedit::implicitSizeBinding_data() { implicitSize_data(); } void tst_qquicktextedit::implicitSizeBinding() { QFETCH(QString, text); QFETCH(QString, wrap); QFETCH(QString, format); QString componentStr = "import QtQuick 2.0\nTextEdit { text: \"" + text + "\"; width: implicitWidth; height: implicitHeight; wrapMode: " + wrap + "; textFormat: " + format + " }"; QDeclarativeComponent textComponent(&engine); textComponent.setData(componentStr.toLatin1(), QUrl::fromLocalFile("")); QScopedPointer object(textComponent.create()); QQuickTextEdit *textObject = qobject_cast(object.data()); QCOMPARE(textObject->width(), textObject->implicitWidth()); QCOMPARE(textObject->height(), textObject->implicitHeight()); textObject->resetWidth(); QCOMPARE(textObject->width(), textObject->implicitWidth()); QCOMPARE(textObject->height(), textObject->implicitHeight()); textObject->resetHeight(); QCOMPARE(textObject->width(), textObject->implicitWidth()); QCOMPARE(textObject->height(), textObject->implicitHeight()); } void tst_qquicktextedit::preeditCursorRectangle() { QString preeditText = "super"; QQuickView view(testFileUrl("inputMethodEvent.qml")); view.show(); view.requestActivateWindow(); QTest::qWaitForWindowShown(&view); QTRY_COMPARE(&view, qGuiApp->focusWindow()); QQuickTextEdit *edit = qobject_cast(view.rootObject()); QVERIFY(edit); QSignalSpy editSpy(edit, SIGNAL(cursorRectangleChanged())); QSignalSpy panelSpy(qGuiApp->inputMethod(), SIGNAL(cursorRectangleChanged())); QRect currentRect; QInputMethodQueryEvent query(Qt::ImCursorRectangle); QCoreApplication::sendEvent(qGuiApp->focusObject(), &query); QRect previousRect = query.value(Qt::ImCursorRectangle).toRect(); // Verify that the micro focus rect is positioned the same for position 0 as // it would be if there was no preedit text. QInputMethodEvent imEvent(preeditText, QList() << QInputMethodEvent::Attribute(QInputMethodEvent::Cursor, 0, preeditText.length(), QVariant())); QCoreApplication::sendEvent(qGuiApp->focusObject(), &imEvent); QCoreApplication::sendEvent(qGuiApp->focusObject(), &query); currentRect = query.value(Qt::ImCursorRectangle).toRect(); QCOMPARE(currentRect, previousRect); QCOMPARE(editSpy.count(), 0); QCOMPARE(panelSpy.count(), 0); // Verify that the micro focus rect moves to the left as the cursor position // is incremented. for (int i = 1; i <= 5; ++i) { QInputMethodEvent imEvent(preeditText, QList() << QInputMethodEvent::Attribute(QInputMethodEvent::Cursor, i, preeditText.length(), QVariant())); QCoreApplication::sendEvent(qGuiApp->focusObject(), &imEvent); QCoreApplication::sendEvent(qGuiApp->focusObject(), &query); currentRect = query.value(Qt::ImCursorRectangle).toRect(); QVERIFY(previousRect.left() < currentRect.left()); QVERIFY(editSpy.count() > 0); editSpy.clear(); QVERIFY(panelSpy.count() > 0); panelSpy.clear(); previousRect = currentRect; } // Verify that if there is no preedit cursor then the micro focus rect is the // same as it would be if it were positioned at the end of the preedit text. QCoreApplication::sendEvent(qGuiApp->focusObject(), &imEvent); editSpy.clear(); panelSpy.clear(); { QInputMethodEvent imEvent(preeditText, QList()); QCoreApplication::sendEvent(qGuiApp->focusObject(), &imEvent); } QCoreApplication::sendEvent(qGuiApp->focusObject(), &query); currentRect = query.value(Qt::ImCursorRectangle).toRect(); QCOMPARE(currentRect, previousRect); QVERIFY(editSpy.count() > 0); QVERIFY(panelSpy.count() > 0); } void tst_qquicktextedit::inputMethodComposing() { QString text = "supercalifragisiticexpialidocious!"; QQuickView view(testFileUrl("inputContext.qml")); view.show(); view.requestActivateWindow(); QTest::qWaitForWindowShown(&view); QTRY_COMPARE(&view, qGuiApp->focusWindow()); QQuickTextEdit *edit = qobject_cast(view.rootObject()); QVERIFY(edit); QSignalSpy spy(edit, SIGNAL(inputMethodComposingChanged())); edit->setCursorPosition(12); QCOMPARE(edit->isInputMethodComposing(), false); { QInputMethodEvent event(text.mid(3), QList()); QGuiApplication::sendEvent(edit, &event); } QCOMPARE(edit->isInputMethodComposing(), true); QCOMPARE(spy.count(), 1); { QInputMethodEvent event(text.mid(12), QList()); QGuiApplication::sendEvent(edit, &event); } QCOMPARE(spy.count(), 1); { QInputMethodEvent event; QGuiApplication::sendEvent(edit, &event); } QCOMPARE(edit->isInputMethodComposing(), false); QCOMPARE(spy.count(), 2); } void tst_qquicktextedit::cursorRectangleSize() { QQuickView *canvas = new QQuickView(testFileUrl("positionAt.qml")); QVERIFY(canvas->rootObject() != 0); QQuickTextEdit *textEdit = qobject_cast(canvas->rootObject()); // make sure cursor rectangle is not at (0,0) textEdit->setX(10); textEdit->setY(10); textEdit->setCursorPosition(3); QVERIFY(textEdit != 0); textEdit->setFocus(true); canvas->show(); canvas->requestActivateWindow(); QTest::qWaitForWindowShown(canvas); QInputMethodQueryEvent event(Qt::ImCursorRectangle); qApp->sendEvent(qApp->focusObject(), &event); QRectF cursorRectFromQuery = event.value(Qt::ImCursorRectangle).toRectF(); QRectF cursorRectFromItem = textEdit->cursorRectangle(); QRectF cursorRectFromPositionToRectangle = textEdit->positionToRectangle(textEdit->cursorPosition()); // item and input query cursor rectangles match QCOMPARE(cursorRectFromItem, cursorRectFromQuery); // item cursor rectangle and positionToRectangle calculations match QCOMPARE(cursorRectFromItem, cursorRectFromPositionToRectangle); // item-canvas transform and input item transform match QCOMPARE(QQuickItemPrivate::get(textEdit)->itemToCanvasTransform(), qApp->inputMethod()->inputItemTransform()); // input panel cursorRectangle property and tranformed item cursor rectangle match QRectF sceneCursorRect = QQuickItemPrivate::get(textEdit)->itemToCanvasTransform().mapRect(cursorRectFromItem); QCOMPARE(sceneCursorRect, qApp->inputMethod()->cursorRectangle()); delete canvas; } void tst_qquicktextedit::getText_data() { QTest::addColumn("text"); QTest::addColumn("start"); QTest::addColumn("end"); QTest::addColumn("expectedText"); const QString richBoldText = QStringLiteral("This is some bold text"); const QString plainBoldText = QStringLiteral("This is some bold text"); QTest::newRow("all plain text") << standard.at(0) << 0 << standard.at(0).length() << standard.at(0); QTest::newRow("plain text sub string") << standard.at(0) << 0 << 12 << standard.at(0).mid(0, 12); QTest::newRow("plain text sub string reversed") << standard.at(0) << 12 << 0 << standard.at(0).mid(0, 12); QTest::newRow("plain text cropped beginning") << standard.at(0) << -3 << 4 << standard.at(0).mid(0, 4); QTest::newRow("plain text cropped end") << standard.at(0) << 23 << standard.at(0).length() + 8 << standard.at(0).mid(23); QTest::newRow("plain text cropped beginning and end") << standard.at(0) << -9 << standard.at(0).length() + 4 << standard.at(0); QTest::newRow("all rich text") << richBoldText << 0 << plainBoldText.length() << plainBoldText; QTest::newRow("rich text sub string") << richBoldText << 14 << 21 << plainBoldText.mid(14, 7); } void tst_qquicktextedit::getText() { QFETCH(QString, text); QFETCH(int, start); QFETCH(int, end); QFETCH(QString, expectedText); QString componentStr = "import QtQuick 2.0\nTextEdit { textFormat: TextEdit.AutoText; text: \"" + text + "\" }"; QQmlComponent textEditComponent(&engine); textEditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEdit = qobject_cast(textEditComponent.create()); QVERIFY(textEdit != 0); QCOMPARE(textEdit->getText(start, end), expectedText); } void tst_qquicktextedit::getFormattedText_data() { QTest::addColumn("text"); QTest::addColumn("textFormat"); QTest::addColumn("start"); QTest::addColumn("end"); QTest::addColumn("expectedText"); const QString richBoldText = QStringLiteral("This is some bold text"); const QString plainBoldText = QStringLiteral("This is some bold text"); QTest::newRow("all plain text") << standard.at(0) << QQuickTextEdit::PlainText << 0 << standard.at(0).length() << standard.at(0); QTest::newRow("plain text sub string") << standard.at(0) << QQuickTextEdit::PlainText << 0 << 12 << standard.at(0).mid(0, 12); QTest::newRow("plain text sub string reversed") << standard.at(0) << QQuickTextEdit::PlainText << 12 << 0 << standard.at(0).mid(0, 12); QTest::newRow("plain text cropped beginning") << standard.at(0) << QQuickTextEdit::PlainText << -3 << 4 << standard.at(0).mid(0, 4); QTest::newRow("plain text cropped end") << standard.at(0) << QQuickTextEdit::PlainText << 23 << standard.at(0).length() + 8 << standard.at(0).mid(23); QTest::newRow("plain text cropped beginning and end") << standard.at(0) << QQuickTextEdit::PlainText << -9 << standard.at(0).length() + 4 << standard.at(0); QTest::newRow("all rich (Auto) text") << richBoldText << QQuickTextEdit::AutoText << 0 << plainBoldText.length() << QString("This is some \\<.*\\>bold\\ text"); QTest::newRow("all rich (Rich) text") << richBoldText << QQuickTextEdit::RichText << 0 << plainBoldText.length() << QString("This is some \\<.*\\>bold\\ text"); QTest::newRow("all rich (Plain) text") << richBoldText << QQuickTextEdit::PlainText << 0 << richBoldText.length() << richBoldText; QTest::newRow("rich (Auto) text sub string") << richBoldText << QQuickTextEdit::AutoText << 14 << 21 << QString("\\<.*\\>old\\ tex"); QTest::newRow("rich (Rich) text sub string") << richBoldText << QQuickTextEdit::RichText << 14 << 21 << QString("\\<.*\\>old\\ tex"); QTest::newRow("rich (Plain) text sub string") << richBoldText << QQuickTextEdit::PlainText << 17 << 27 << richBoldText.mid(17, 10); } void tst_qquicktextedit::getFormattedText() { QFETCH(QString, text); QFETCH(QQuickTextEdit::TextFormat, textFormat); QFETCH(int, start); QFETCH(int, end); QFETCH(QString, expectedText); QString componentStr = "import QtQuick 2.0\nTextEdit {}"; QQmlComponent textEditComponent(&engine); textEditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEdit = qobject_cast(textEditComponent.create()); QVERIFY(textEdit != 0); textEdit->setTextFormat(textFormat); textEdit->setText(text); if (textFormat == QQuickTextEdit::RichText || (textFormat == QQuickTextEdit::AutoText && Qt::mightBeRichText(text))) { QVERIFY(textEdit->getFormattedText(start, end).contains(QRegExp(expectedText))); } else { QCOMPARE(textEdit->getFormattedText(start, end), expectedText); } } void tst_qquicktextedit::insert_data() { QTest::addColumn("text"); QTest::addColumn("textFormat"); QTest::addColumn("selectionStart"); QTest::addColumn("selectionEnd"); QTest::addColumn("insertPosition"); QTest::addColumn("insertText"); QTest::addColumn("expectedText"); QTest::addColumn("expectedSelectionStart"); QTest::addColumn("expectedSelectionEnd"); QTest::addColumn("expectedCursorPosition"); QTest::addColumn("selectionChanged"); QTest::addColumn("cursorPositionChanged"); QTest::newRow("at cursor position (beginning)") << standard.at(0) << QQuickTextEdit::PlainText << 0 << 0 << 0 << QString("Hello") << QString("Hello") + standard.at(0) << 5 << 5 << 5 << false << true; QTest::newRow("at cursor position (end)") << standard.at(0) << QQuickTextEdit::PlainText << standard.at(0).length() << standard.at(0).length() << standard.at(0).length() << QString("Hello") << standard.at(0) + QString("Hello") << standard.at(0).length() + 5 << standard.at(0).length() + 5 << standard.at(0).length() + 5 << false << true; QTest::newRow("at cursor position (middle)") << standard.at(0) << QQuickTextEdit::PlainText << 18 << 18 << 18 << QString("Hello") << standard.at(0).mid(0, 18) + QString("Hello") + standard.at(0).mid(18) << 23 << 23 << 23 << false << true; QTest::newRow("after cursor position (beginning)") << standard.at(0) << QQuickTextEdit::PlainText << 0 << 0 << 18 << QString("Hello") << standard.at(0).mid(0, 18) + QString("Hello") + standard.at(0).mid(18) << 0 << 0 << 0 << false << false; QTest::newRow("before cursor position (end)") << standard.at(0) << QQuickTextEdit::PlainText << standard.at(0).length() << standard.at(0).length() << 18 << QString("Hello") << standard.at(0).mid(0, 18) + QString("Hello") + standard.at(0).mid(18) << standard.at(0).length() + 5 << standard.at(0).length() + 5 << standard.at(0).length() + 5 << false << true; QTest::newRow("before cursor position (middle)") << standard.at(0) << QQuickTextEdit::PlainText << 18 << 18 << 0 << QString("Hello") << QString("Hello") + standard.at(0) << 23 << 23 << 23 << false << true; QTest::newRow("after cursor position (middle)") << standard.at(0) << QQuickTextEdit::PlainText << 18 << 18 << standard.at(0).length() << QString("Hello") << standard.at(0) + QString("Hello") << 18 << 18 << 18 << false << false; QTest::newRow("before selection") << standard.at(0) << QQuickTextEdit::PlainText << 14 << 19 << 0 << QString("Hello") << QString("Hello") + standard.at(0) << 19 << 24 << 24 << false << true; QTest::newRow("before reversed selection") << standard.at(0) << QQuickTextEdit::PlainText << 19 << 14 << 0 << QString("Hello") << QString("Hello") + standard.at(0) << 19 << 24 << 19 << false << true; QTest::newRow("after selection") << standard.at(0) << QQuickTextEdit::PlainText << 14 << 19 << standard.at(0).length() << QString("Hello") << standard.at(0) + QString("Hello") << 14 << 19 << 19 << false << false; QTest::newRow("after reversed selection") << standard.at(0) << QQuickTextEdit::PlainText << 19 << 14 << standard.at(0).length() << QString("Hello") << standard.at(0) + QString("Hello") << 14 << 19 << 14 << false << false; QTest::newRow("into selection") << standard.at(0) << QQuickTextEdit::PlainText << 14 << 19 << 18 << QString("Hello") << standard.at(0).mid(0, 18) + QString("Hello") + standard.at(0).mid(18) << 14 << 24 << 24 << true << true; QTest::newRow("into reversed selection") << standard.at(0) << QQuickTextEdit::PlainText << 19 << 14 << 18 << QString("Hello") << standard.at(0).mid(0, 18) + QString("Hello") + standard.at(0).mid(18) << 14 << 24 << 14 << true << false; QTest::newRow("rich text into plain text") << standard.at(0) << QQuickTextEdit::PlainText << 0 << 0 << 0 << QString("Hello") << QString("Hello") + standard.at(0) << 12 << 12 << 12 << false << true; QTest::newRow("rich text into rich text") << standard.at(0) << QQuickTextEdit::RichText << 0 << 0 << 0 << QString("Hello") << QString("Hello") + standard.at(0) << 5 << 5 << 5 << false << true; QTest::newRow("rich text into auto text") << standard.at(0) << QQuickTextEdit::AutoText << 0 << 0 << 0 << QString("Hello") << QString("Hello") + standard.at(0) << 5 << 5 << 5 << false << true; QTest::newRow("before start") << standard.at(0) << QQuickTextEdit::PlainText << 0 << 0 << -3 << QString("Hello") << standard.at(0) << 0 << 0 << 0 << false << false; QTest::newRow("past end") << standard.at(0) << QQuickTextEdit::PlainText << 0 << 0 << standard.at(0).length() + 3 << QString("Hello") << standard.at(0) << 0 << 0 << 0 << false << false; } void tst_qquicktextedit::insert() { QFETCH(QString, text); QFETCH(QQuickTextEdit::TextFormat, textFormat); QFETCH(int, selectionStart); QFETCH(int, selectionEnd); QFETCH(int, insertPosition); QFETCH(QString, insertText); QFETCH(QString, expectedText); QFETCH(int, expectedSelectionStart); QFETCH(int, expectedSelectionEnd); QFETCH(int, expectedCursorPosition); QFETCH(bool, selectionChanged); QFETCH(bool, cursorPositionChanged); QString componentStr = "import QtQuick 2.0\nTextEdit { text: \"" + text + "\" }"; QQmlComponent textEditComponent(&engine); textEditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEdit = qobject_cast(textEditComponent.create()); QVERIFY(textEdit != 0); textEdit->setTextFormat(textFormat); textEdit->select(selectionStart, selectionEnd); QSignalSpy selectionSpy(textEdit, SIGNAL(selectionChanged())); QSignalSpy selectionStartSpy(textEdit, SIGNAL(selectionStartChanged())); QSignalSpy selectionEndSpy(textEdit, SIGNAL(selectionEndChanged())); QSignalSpy textSpy(textEdit, SIGNAL(textChanged())); QSignalSpy cursorPositionSpy(textEdit, SIGNAL(cursorPositionChanged())); textEdit->insert(insertPosition, insertText); if (textFormat == QQuickTextEdit::RichText || (textFormat == QQuickTextEdit::AutoText && ( Qt::mightBeRichText(text) || Qt::mightBeRichText(insertText)))) { QCOMPARE(textEdit->getText(0, expectedText.length()), expectedText); } else { QCOMPARE(textEdit->text(), expectedText); } QCOMPARE(textEdit->length(), expectedText.length()); QCOMPARE(textEdit->selectionStart(), expectedSelectionStart); QCOMPARE(textEdit->selectionEnd(), expectedSelectionEnd); QCOMPARE(textEdit->cursorPosition(), expectedCursorPosition); if (selectionStart > selectionEnd) qSwap(selectionStart, selectionEnd); QEXPECT_FAIL("into selection", "selectionChanged signal isn't emitted on edits within selection", Continue); QEXPECT_FAIL("into reversed selection", "selectionChanged signal isn't emitted on edits within selection", Continue); QCOMPARE(selectionSpy.count() > 0, selectionChanged); QCOMPARE(selectionStartSpy.count() > 0, selectionStart != expectedSelectionStart); QEXPECT_FAIL("into reversed selection", "selectionEndChanged signal not emitted", Continue); QCOMPARE(selectionEndSpy.count() > 0, selectionEnd != expectedSelectionEnd); QCOMPARE(textSpy.count() > 0, text != expectedText); QCOMPARE(cursorPositionSpy.count() > 0, cursorPositionChanged); } void tst_qquicktextedit::remove_data() { QTest::addColumn("text"); QTest::addColumn("textFormat"); QTest::addColumn("selectionStart"); QTest::addColumn("selectionEnd"); QTest::addColumn("removeStart"); QTest::addColumn("removeEnd"); QTest::addColumn("expectedText"); QTest::addColumn("expectedSelectionStart"); QTest::addColumn("expectedSelectionEnd"); QTest::addColumn("expectedCursorPosition"); QTest::addColumn("selectionChanged"); QTest::addColumn("cursorPositionChanged"); const QString richBoldText = QStringLiteral("This is some bold text"); const QString plainBoldText = QStringLiteral("This is some bold text"); QTest::newRow("from cursor position (beginning)") << standard.at(0) << QQuickTextEdit::PlainText << 0 << 0 << 0 << 5 << standard.at(0).mid(5) << 0 << 0 << 0 << false << false; QTest::newRow("to cursor position (beginning)") << standard.at(0) << QQuickTextEdit::PlainText << 0 << 0 << 5 << 0 << standard.at(0).mid(5) << 0 << 0 << 0 << false << false; QTest::newRow("to cursor position (end)") << standard.at(0) << QQuickTextEdit::PlainText << standard.at(0).length() << standard.at(0).length() << standard.at(0).length() << standard.at(0).length() - 5 << standard.at(0).mid(0, standard.at(0).length() - 5) << standard.at(0).length() - 5 << standard.at(0).length() - 5 << standard.at(0).length() - 5 << false << true; QTest::newRow("to cursor position (end)") << standard.at(0) << QQuickTextEdit::PlainText << standard.at(0).length() << standard.at(0).length() << standard.at(0).length() - 5 << standard.at(0).length() << standard.at(0).mid(0, standard.at(0).length() - 5) << standard.at(0).length() - 5 << standard.at(0).length() - 5 << standard.at(0).length() - 5 << false << true; QTest::newRow("from cursor position (middle)") << standard.at(0) << QQuickTextEdit::PlainText << 18 << 18 << 18 << 23 << standard.at(0).mid(0, 18) + standard.at(0).mid(23) << 18 << 18 << 18 << false << false; QTest::newRow("to cursor position (middle)") << standard.at(0) << QQuickTextEdit::PlainText << 23 << 23 << 18 << 23 << standard.at(0).mid(0, 18) + standard.at(0).mid(23) << 18 << 18 << 18 << false << true; QTest::newRow("after cursor position (beginning)") << standard.at(0) << QQuickTextEdit::PlainText << 0 << 0 << 18 << 23 << standard.at(0).mid(0, 18) + standard.at(0).mid(23) << 0 << 0 << 0 << false << false; QTest::newRow("before cursor position (end)") << standard.at(0) << QQuickTextEdit::PlainText << standard.at(0).length() << standard.at(0).length() << 18 << 23 << standard.at(0).mid(0, 18) + standard.at(0).mid(23) << standard.at(0).length() - 5 << standard.at(0).length() - 5 << standard.at(0).length() - 5 << false << true; QTest::newRow("before cursor position (middle)") << standard.at(0) << QQuickTextEdit::PlainText << 23 << 23 << 0 << 5 << standard.at(0).mid(5) << 18 << 18 << 18 << false << true; QTest::newRow("after cursor position (middle)") << standard.at(0) << QQuickTextEdit::PlainText << 18 << 18 << 18 << 23 << standard.at(0).mid(0, 18) + standard.at(0).mid(23) << 18 << 18 << 18 << false << false; QTest::newRow("before selection") << standard.at(0) << QQuickTextEdit::PlainText << 14 << 19 << 0 << 5 << standard.at(0).mid(5) << 9 << 14 << 14 << false << true; QTest::newRow("before reversed selection") << standard.at(0) << QQuickTextEdit::PlainText << 19 << 14 << 0 << 5 << standard.at(0).mid(5) << 9 << 14 << 9 << false << true; QTest::newRow("after selection") << standard.at(0) << QQuickTextEdit::PlainText << 14 << 19 << standard.at(0).length() - 5 << standard.at(0).length() << standard.at(0).mid(0, standard.at(0).length() - 5) << 14 << 19 << 19 << false << false; QTest::newRow("after reversed selection") << standard.at(0) << QQuickTextEdit::PlainText << 19 << 14 << standard.at(0).length() - 5 << standard.at(0).length() << standard.at(0).mid(0, standard.at(0).length() - 5) << 14 << 19 << 14 << false << false; QTest::newRow("from selection") << standard.at(0) << QQuickTextEdit::PlainText << 14 << 24 << 18 << 23 << standard.at(0).mid(0, 18) + standard.at(0).mid(23) << 14 << 19 << 19 << true << true; QTest::newRow("from reversed selection") << standard.at(0) << QQuickTextEdit::PlainText << 24 << 14 << 18 << 23 << standard.at(0).mid(0, 18) + standard.at(0).mid(23) << 14 << 19 << 14 << true << false; QTest::newRow("plain text cropped beginning") << standard.at(0) << QQuickTextEdit::PlainText << 0 << 0 << -3 << 4 << standard.at(0).mid(4) << 0 << 0 << 0 << false << false; QTest::newRow("plain text cropped end") << standard.at(0) << QQuickTextEdit::PlainText << 0 << 0 << 23 << standard.at(0).length() + 8 << standard.at(0).mid(0, 23) << 0 << 0 << 0 << false << false; QTest::newRow("plain text cropped beginning and end") << standard.at(0) << QQuickTextEdit::PlainText << 0 << 0 << -9 << standard.at(0).length() + 4 << QString() << 0 << 0 << 0 << false << false; QTest::newRow("all rich text") << richBoldText << QQuickTextEdit::RichText << 0 << 0 << 0 << plainBoldText.length() << QString() << 0 << 0 << 0 << false << false; QTest::newRow("rick text sub string") << richBoldText << QQuickTextEdit::RichText << 0 << 0 << 14 << 21 << plainBoldText.mid(0, 14) + plainBoldText.mid(21) << 0 << 0 << 0 << false << false; } void tst_qquicktextedit::remove() { QFETCH(QString, text); QFETCH(QQuickTextEdit::TextFormat, textFormat); QFETCH(int, selectionStart); QFETCH(int, selectionEnd); QFETCH(int, removeStart); QFETCH(int, removeEnd); QFETCH(QString, expectedText); QFETCH(int, expectedSelectionStart); QFETCH(int, expectedSelectionEnd); QFETCH(int, expectedCursorPosition); QFETCH(bool, selectionChanged); QFETCH(bool, cursorPositionChanged); QString componentStr = "import QtQuick 2.0\nTextEdit { text: \"" + text + "\" }"; QQmlComponent textEditComponent(&engine); textEditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEdit = qobject_cast(textEditComponent.create()); QVERIFY(textEdit != 0); textEdit->setTextFormat(textFormat); textEdit->select(selectionStart, selectionEnd); QSignalSpy selectionSpy(textEdit, SIGNAL(selectionChanged())); QSignalSpy selectionStartSpy(textEdit, SIGNAL(selectionStartChanged())); QSignalSpy selectionEndSpy(textEdit, SIGNAL(selectionEndChanged())); QSignalSpy textSpy(textEdit, SIGNAL(textChanged())); QSignalSpy cursorPositionSpy(textEdit, SIGNAL(cursorPositionChanged())); textEdit->remove(removeStart, removeEnd); if (textFormat == QQuickTextEdit::RichText || (textFormat == QQuickTextEdit::AutoText && Qt::mightBeRichText(text))) { QCOMPARE(textEdit->getText(0, expectedText.length()), expectedText); } else { QCOMPARE(textEdit->text(), expectedText); } QCOMPARE(textEdit->length(), expectedText.length()); if (selectionStart > selectionEnd) // qSwap(selectionStart, selectionEnd); QCOMPARE(textEdit->selectionStart(), expectedSelectionStart); QCOMPARE(textEdit->selectionEnd(), expectedSelectionEnd); QCOMPARE(textEdit->cursorPosition(), expectedCursorPosition); QEXPECT_FAIL("from selection", "selectionChanged signal isn't emitted on edits within selection", Continue); QEXPECT_FAIL("from reversed selection", "selectionChanged signal isn't emitted on edits within selection", Continue); QCOMPARE(selectionSpy.count() > 0, selectionChanged); QCOMPARE(selectionStartSpy.count() > 0, selectionStart != expectedSelectionStart); QEXPECT_FAIL("from reversed selection", "selectionEndChanged signal not emitted", Continue); QCOMPARE(selectionEndSpy.count() > 0, selectionEnd != expectedSelectionEnd); QCOMPARE(textSpy.count() > 0, text != expectedText); if (cursorPositionChanged) // QVERIFY(cursorPositionSpy.count() > 0); } void tst_qquicktextedit::keySequence_data() { QTest::addColumn("text"); QTest::addColumn("sequence"); QTest::addColumn("selectionStart"); QTest::addColumn("selectionEnd"); QTest::addColumn("cursorPosition"); QTest::addColumn("expectedText"); QTest::addColumn("selectedText"); QTest::addColumn("layoutDirection"); // standard[0] == "the [4]quick [10]brown [16]fox [20]jumped [27]over [32]the [36]lazy [41]dog" QTest::newRow("select all") << standard.at(0) << QKeySequence(QKeySequence::SelectAll) << 0 << 0 << 44 << standard.at(0) << standard.at(0) << Qt::Key_Direction_L; QTest::newRow("select start of line") << standard.at(0) << QKeySequence(QKeySequence::SelectStartOfLine) << 5 << 5 << 0 << standard.at(0) << standard.at(0).mid(0, 5) << Qt::Key_Direction_L; QTest::newRow("select start of block") << standard.at(0) << QKeySequence(QKeySequence::SelectStartOfBlock) << 5 << 5 << 0 << standard.at(0) << standard.at(0).mid(0, 5) << Qt::Key_Direction_L; QTest::newRow("select end of line") << standard.at(0) << QKeySequence(QKeySequence::SelectEndOfLine) << 5 << 5 << 44 << standard.at(0) << standard.at(0).mid(5) << Qt::Key_Direction_L; QTest::newRow("select end of document") << standard.at(0) << QKeySequence(QKeySequence::SelectEndOfDocument) << 3 << 3 << 44 << standard.at(0) << standard.at(0).mid(3) << Qt::Key_Direction_L; QTest::newRow("select end of block") << standard.at(0) << QKeySequence(QKeySequence::SelectEndOfBlock) << 18 << 18 << 44 << standard.at(0) << standard.at(0).mid(18) << Qt::Key_Direction_L; QTest::newRow("delete end of line") << standard.at(0) << QKeySequence(QKeySequence::DeleteEndOfLine) << 24 << 24 << 24 << standard.at(0).mid(0, 24) << QString() << Qt::Key_Direction_L; QTest::newRow("move to start of line") << standard.at(0) << QKeySequence(QKeySequence::MoveToStartOfLine) << 31 << 31 << 0 << standard.at(0) << QString() << Qt::Key_Direction_L; QTest::newRow("move to start of block") << standard.at(0) << QKeySequence(QKeySequence::MoveToStartOfBlock) << 25 << 25 << 0 << standard.at(0) << QString() << Qt::Key_Direction_L; QTest::newRow("move to next char") << standard.at(0) << QKeySequence(QKeySequence::MoveToNextChar) << 12 << 12 << 13 << standard.at(0) << QString() << Qt::Key_Direction_L; QTest::newRow("move to previous char (ltr)") << standard.at(0) << QKeySequence(QKeySequence::MoveToPreviousChar) << 3 << 3 << 2 << standard.at(0) << QString() << Qt::Key_Direction_L; QTest::newRow("move to previous char (rtl)") << standard.at(0) << QKeySequence(QKeySequence::MoveToPreviousChar) << 3 << 3 << 4 << standard.at(0) << QString() << Qt::Key_Direction_R; QTest::newRow("move to previous char with selection") << standard.at(0) << QKeySequence(QKeySequence::MoveToPreviousChar) << 3 << 7 << 3 << standard.at(0) << QString() << Qt::Key_Direction_L; QTest::newRow("select next char (ltr)") << standard.at(0) << QKeySequence(QKeySequence::SelectNextChar) << 23 << 23 << 24 << standard.at(0) << standard.at(0).mid(23, 1) << Qt::Key_Direction_L; QTest::newRow("select next char (rtl)") << standard.at(0) << QKeySequence(QKeySequence::SelectNextChar) << 23 << 23 << 22 << standard.at(0) << standard.at(0).mid(22, 1) << Qt::Key_Direction_R; QTest::newRow("select previous char (ltr)") << standard.at(0) << QKeySequence(QKeySequence::SelectPreviousChar) << 19 << 19 << 18 << standard.at(0) << standard.at(0).mid(18, 1) << Qt::Key_Direction_L; QTest::newRow("select previous char (rtl)") << standard.at(0) << QKeySequence(QKeySequence::SelectPreviousChar) << 19 << 19 << 20 << standard.at(0) << standard.at(0).mid(19, 1) << Qt::Key_Direction_R; QTest::newRow("move to next word (ltr)") << standard.at(0) << QKeySequence(QKeySequence::MoveToNextWord) << 7 << 7 << 10 << standard.at(0) << QString() << Qt::Key_Direction_L; QTest::newRow("move to next word (rtl)") << standard.at(0) << QKeySequence(QKeySequence::MoveToNextWord) << 7 << 7 << 4 << standard.at(0) << QString() << Qt::Key_Direction_R; QTest::newRow("move to previous word (ltr)") << standard.at(0) << QKeySequence(QKeySequence::MoveToPreviousWord) << 7 << 7 << 4 << standard.at(0) << QString() << Qt::Key_Direction_L; QTest::newRow("move to previous word (rlt)") << standard.at(0) << QKeySequence(QKeySequence::MoveToPreviousWord) << 7 << 7 << 10 << standard.at(0) << QString() << Qt::Key_Direction_R; QTest::newRow("select next word") << standard.at(0) << QKeySequence(QKeySequence::SelectNextWord) << 11 << 11 << 16 << standard.at(0) << standard.at(0).mid(11, 5) << Qt::Key_Direction_L; QTest::newRow("select previous word") << standard.at(0) << QKeySequence(QKeySequence::SelectPreviousWord) << 11 << 11 << 10 << standard.at(0) << standard.at(0).mid(10, 1) << Qt::Key_Direction_L; QTest::newRow("delete (selection)") << standard.at(0) << QKeySequence(QKeySequence::Delete) << 12 << 15 << 12 << (standard.at(0).mid(0, 12) + standard.at(0).mid(15)) << QString() << Qt::Key_Direction_L; QTest::newRow("delete (no selection)") << standard.at(0) << QKeySequence(QKeySequence::Delete) << 15 << 15 << 15 << (standard.at(0).mid(0, 15) + standard.at(0).mid(16)) << QString() << Qt::Key_Direction_L; QTest::newRow("delete end of word") << standard.at(0) << QKeySequence(QKeySequence::DeleteEndOfWord) << 24 << 24 << 24 << (standard.at(0).mid(0, 24) + standard.at(0).mid(27)) << QString() << Qt::Key_Direction_L; QTest::newRow("delete start of word") << standard.at(0) << QKeySequence(QKeySequence::DeleteStartOfWord) << 7 << 7 << 4 << (standard.at(0).mid(0, 4) + standard.at(0).mid(7)) << QString() << Qt::Key_Direction_L; } void tst_qquicktextedit::keySequence() { QFETCH(QString, text); QFETCH(QKeySequence, sequence); QFETCH(int, selectionStart); QFETCH(int, selectionEnd); QFETCH(int, cursorPosition); QFETCH(QString, expectedText); QFETCH(QString, selectedText); QFETCH(Qt::Key, layoutDirection); if (sequence.isEmpty()) { QSKIP("Key sequence is undefined"); } QString componentStr = "import QtQuick 2.0\nTextEdit { focus: true; text: \"" + text + "\" }"; QQmlComponent textEditComponent(&engine); textEditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEdit = qobject_cast(textEditComponent.create()); QVERIFY(textEdit != 0); QQuickCanvas canvas; textEdit->setParentItem(canvas.rootItem()); canvas.show(); canvas.requestActivateWindow(); QTest::qWaitForWindowShown(&canvas); QTRY_COMPARE(QGuiApplication::activeWindow(), &canvas); simulateKey(&canvas, layoutDirection); textEdit->select(selectionStart, selectionEnd); simulateKeys(&canvas, sequence); QCOMPARE(textEdit->cursorPosition(), cursorPosition); QCOMPARE(textEdit->text(), expectedText); QCOMPARE(textEdit->selectedText(), selectedText); } #define NORMAL 0 #define REPLACE_UNTIL_END 1 void tst_qquicktextedit::undo_data() { QTest::addColumn("insertString"); QTest::addColumn("insertIndex"); QTest::addColumn("insertMode"); QTest::addColumn("expectedString"); QTest::addColumn("use_keys"); for (int i=0; i<2; i++) { QString keys_str = "keyboard"; bool use_keys = true; if (i==0) { keys_str = "insert"; use_keys = false; } { IntList insertIndex; IntList insertMode; QStringList insertString; QStringList expectedString; insertIndex << -1; insertMode << NORMAL; insertString << "1"; insertIndex << -1; insertMode << NORMAL; insertString << "5"; insertIndex << 1; insertMode << NORMAL; insertString << "3"; insertIndex << 1; insertMode << NORMAL; insertString << "2"; insertIndex << 3; insertMode << NORMAL; insertString << "4"; expectedString << "12345"; expectedString << "1235"; expectedString << "135"; expectedString << "15"; expectedString << ""; QTest::newRow(QString(keys_str + "_numbers").toLatin1()) << insertString << insertIndex << insertMode << expectedString << bool(use_keys); } { IntList insertIndex; IntList insertMode; QStringList insertString; QStringList expectedString; insertIndex << -1; insertMode << NORMAL; insertString << "World"; // World insertIndex << 0; insertMode << NORMAL; insertString << "Hello"; // HelloWorld insertIndex << 0; insertMode << NORMAL; insertString << "Well"; // WellHelloWorld insertIndex << 9; insertMode << NORMAL; insertString << "There"; // WellHelloThereWorld; expectedString << "WellHelloThereWorld"; expectedString << "WellHelloWorld"; expectedString << "HelloWorld"; expectedString << "World"; expectedString << ""; QTest::newRow(QString(keys_str + "_helloworld").toLatin1()) << insertString << insertIndex << insertMode << expectedString << bool(use_keys); } { IntList insertIndex; IntList insertMode; QStringList insertString; QStringList expectedString; insertIndex << -1; insertMode << NORMAL; insertString << "Ensuring"; insertIndex << -1; insertMode << NORMAL; insertString << " instan"; insertIndex << 9; insertMode << NORMAL; insertString << "an "; insertIndex << 10; insertMode << REPLACE_UNTIL_END; insertString << " unique instance."; expectedString << "Ensuring a unique instance."; expectedString << "Ensuring a "; // ### Not present in TextEdit. expectedString << "Ensuring an instan"; expectedString << "Ensuring instan"; expectedString << ""; QTest::newRow(QString(keys_str + "_patterns").toLatin1()) << insertString << insertIndex << insertMode << expectedString << bool(use_keys); } } } void tst_qquicktextedit::undo() { QFETCH(QStringList, insertString); QFETCH(IntList, insertIndex); QFETCH(IntList, insertMode); QFETCH(QStringList, expectedString); QString componentStr = "import QtQuick 2.0\nTextEdit { focus: true }"; QQmlComponent textEditComponent(&engine); textEditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEdit = qobject_cast(textEditComponent.create()); QVERIFY(textEdit != 0); QQuickCanvas canvas; textEdit->setParentItem(canvas.rootItem()); canvas.show(); canvas.requestActivateWindow(); QTest::qWaitForWindowShown(&canvas); QTRY_COMPARE(QGuiApplication::activeWindow(), &canvas); QVERIFY(!textEdit->canUndo()); QSignalSpy spy(textEdit, SIGNAL(canUndoChanged())); int i; // STEP 1: First build up an undo history by inserting or typing some strings... for (i = 0; i < insertString.size(); ++i) { if (insertIndex[i] > -1) textEdit->setCursorPosition(insertIndex[i]); // experimental stuff if (insertMode[i] == REPLACE_UNTIL_END) { textEdit->select(insertIndex[i], insertIndex[i] + 8); // This is what I actually want... // QTest::keyClick(testWidget, Qt::Key_End, Qt::ShiftModifier); } for (int j = 0; j < insertString.at(i).length(); j++) QTest::keyClick(&canvas, insertString.at(i).at(j).toLatin1()); } QCOMPARE(spy.count(), 1); // STEP 2: Next call undo several times and see if we can restore to the previous state for (i = 0; i < expectedString.size() - 1; ++i) { QCOMPARE(textEdit->text(), expectedString[i]); QVERIFY(textEdit->canUndo()); textEdit->undo(); } // STEP 3: Verify that we have undone everything QVERIFY(textEdit->text().isEmpty()); QVERIFY(!textEdit->canUndo()); QCOMPARE(spy.count(), 2); } void tst_qquicktextedit::redo_data() { QTest::addColumn("insertString"); QTest::addColumn("insertIndex"); QTest::addColumn("expectedString"); { IntList insertIndex; QStringList insertString; QStringList expectedString; insertIndex << -1; insertString << "World"; // World insertIndex << 0; insertString << "Hello"; // HelloWorld insertIndex << 0; insertString << "Well"; // WellHelloWorld insertIndex << 9; insertString << "There"; // WellHelloThereWorld; expectedString << "World"; expectedString << "HelloWorld"; expectedString << "WellHelloWorld"; expectedString << "WellHelloThereWorld"; QTest::newRow("Inserts and setting cursor") << insertString << insertIndex << expectedString; } } void tst_qquicktextedit::redo() { QFETCH(QStringList, insertString); QFETCH(IntList, insertIndex); QFETCH(QStringList, expectedString); QString componentStr = "import QtQuick 2.0\nTextEdit { focus: true }"; QQmlComponent textEditComponent(&engine); textEditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEdit = qobject_cast(textEditComponent.create()); QVERIFY(textEdit != 0); QQuickCanvas canvas; textEdit->setParentItem(canvas.rootItem()); canvas.show(); canvas.requestActivateWindow(); QTest::qWaitForWindowShown(&canvas); QTRY_COMPARE(QGuiApplication::activeWindow(), &canvas); QVERIFY(!textEdit->canUndo()); QVERIFY(!textEdit->canRedo()); QSignalSpy spy(textEdit, SIGNAL(canRedoChanged())); int i; // inserts the diff strings at diff positions for (i = 0; i < insertString.size(); ++i) { if (insertIndex[i] > -1) textEdit->setCursorPosition(insertIndex[i]); for (int j = 0; j < insertString.at(i).length(); j++) QTest::keyClick(&canvas, insertString.at(i).at(j).toLatin1()); QVERIFY(textEdit->canUndo()); QVERIFY(!textEdit->canRedo()); } QCOMPARE(spy.count(), 0); // undo everything while (!textEdit->text().isEmpty()) { QVERIFY(textEdit->canUndo()); textEdit->undo(); QVERIFY(textEdit->canRedo()); } QCOMPARE(spy.count(), 1); for (i = 0; i < expectedString.size(); ++i) { QVERIFY(textEdit->canRedo()); textEdit->redo(); QCOMPARE(textEdit->text() , expectedString[i]); QVERIFY(textEdit->canUndo()); } QVERIFY(!textEdit->canRedo()); QCOMPARE(spy.count(), 2); } void tst_qquicktextedit::undo_keypressevents_data() { QTest::addColumn("keys"); QTest::addColumn("expectedString"); { KeyList keys; QStringList expectedString; keys << "AFRAID" << Qt::Key_Home << "VERY" << Qt::Key_Left << Qt::Key_Left << Qt::Key_Left << Qt::Key_Left << "BE" << Qt::Key_End << "!"; expectedString << "BEVERYAFRAID!"; expectedString << "BEVERYAFRAID"; expectedString << "VERYAFRAID"; expectedString << "AFRAID"; QTest::newRow("Inserts and moving cursor") << keys << expectedString; } { KeyList keys; QStringList expectedString; // inserting '1234' keys << "1234" << Qt::Key_Home // skipping '12' << Qt::Key_Right << Qt::Key_Right // selecting '34' << (Qt::Key_Right | Qt::ShiftModifier) << (Qt::Key_Right | Qt::ShiftModifier) // deleting '34' << Qt::Key_Delete; expectedString << "12"; expectedString << "1234"; QTest::newRow("Inserts,moving,selection and delete") << keys << expectedString; } { KeyList keys; QStringList expectedString; // inserting 'AB12' keys << "AB12" << Qt::Key_Home // selecting 'AB' << (Qt::Key_Right | Qt::ShiftModifier) << (Qt::Key_Right | Qt::ShiftModifier) << Qt::Key_Delete << QKeySequence::Undo // ### Text is selected in text input // << Qt::Key_Right << (Qt::Key_Right | Qt::ShiftModifier) << (Qt::Key_Right | Qt::ShiftModifier) << Qt::Key_Delete; expectedString << "AB"; expectedString << "AB12"; QTest::newRow("Inserts,moving,selection, delete and undo") << keys << expectedString; } { KeyList keys; QStringList expectedString; // inserting 'ABCD' keys << "abcd" //move left two << Qt::Key_Left << Qt::Key_Left // inserting '1234' << "1234" // selecting '1234' << (Qt::Key_Left | Qt::ShiftModifier) << (Qt::Key_Left | Qt::ShiftModifier) << (Qt::Key_Left | Qt::ShiftModifier) << (Qt::Key_Left | Qt::ShiftModifier) // overwriting '1234' with '5' << "5" // undoing deletion of 'AB' << QKeySequence::Undo // ### Text is selected in text input << (Qt::Key_Left | Qt::ShiftModifier) << (Qt::Key_Left | Qt::ShiftModifier) << (Qt::Key_Left | Qt::ShiftModifier) << (Qt::Key_Left | Qt::ShiftModifier) // overwriting '1234' with '6' << "6"; expectedString << "ab6cd"; // for versions previous to 3.2 we overwrite needed two undo operations expectedString << "ab1234cd"; expectedString << "abcd"; QTest::newRow("Inserts,moving,selection and undo, removing selection") << keys << expectedString; } { KeyList keys; QStringList expectedString; // inserting 'ABC' keys << "ABC" // removes 'C' << Qt::Key_Backspace; expectedString << "AB"; expectedString << "ABC"; QTest::newRow("Inserts,backspace") << keys << expectedString; } { KeyList keys; QStringList expectedString; keys << "ABC" // removes 'C' << Qt::Key_Backspace // inserting 'Z' << "Z"; expectedString << "ABZ"; expectedString << "AB"; expectedString << "ABC"; QTest::newRow("Inserts,backspace,inserts") << keys << expectedString; } { KeyList keys; QStringList expectedString; // inserting '123' keys << "123" << Qt::Key_Home // selecting '123' << (Qt::Key_End | Qt::ShiftModifier) // overwriting '123' with 'ABC' << "ABC"; expectedString << "ABC"; // ### One operation in TextEdit. expectedString << "A"; expectedString << "123"; QTest::newRow("Inserts,moving,selection and overwriting") << keys << expectedString; } } void tst_qquicktextedit::undo_keypressevents() { QFETCH(KeyList, keys); QFETCH(QStringList, expectedString); QString componentStr = "import QtQuick 2.0\nTextEdit { focus: true }"; QQmlComponent textEditComponent(&engine); textEditComponent.setData(componentStr.toLatin1(), QUrl()); QQuickTextEdit *textEdit = qobject_cast(textEditComponent.create()); QVERIFY(textEdit != 0); QQuickCanvas canvas; textEdit->setParentItem(canvas.rootItem()); canvas.show(); canvas.requestActivateWindow(); QTest::qWaitForWindowShown(&canvas); QTRY_COMPARE(QGuiApplication::activeWindow(), &canvas); simulateKeys(&canvas, keys); for (int i = 0; i < expectedString.size(); ++i) { QCOMPARE(textEdit->text() , expectedString[i]); textEdit->undo(); } QVERIFY(textEdit->text().isEmpty()); } void tst_qquicktextedit::baseUrl() { QUrl localUrl("file:///tests/text.qml"); QUrl remoteUrl("http://qt.nokia.com/test.qml"); QQmlComponent textComponent(&engine); textComponent.setData("import QtQuick 2.0\n TextEdit {}", localUrl); QQuickTextEdit *textObject = qobject_cast(textComponent.create()); QCOMPARE(textObject->baseUrl(), localUrl); QSignalSpy spy(textObject, SIGNAL(baseUrlChanged())); textObject->setBaseUrl(localUrl); QCOMPARE(textObject->baseUrl(), localUrl); QCOMPARE(spy.count(), 0); textObject->setBaseUrl(remoteUrl); QCOMPARE(textObject->baseUrl(), remoteUrl); QCOMPARE(spy.count(), 1); textObject->resetBaseUrl(); QCOMPARE(textObject->baseUrl(), localUrl); QCOMPARE(spy.count(), 2); } void tst_qquicktextedit::embeddedImages_data() { QTest::addColumn("qmlfile"); QTest::addColumn("error"); QTest::newRow("local") << testFileUrl("embeddedImagesLocal.qml") << ""; QTest::newRow("local-error") << testFileUrl("embeddedImagesLocalError.qml") << testFileUrl("embeddedImagesLocalError.qml").toString()+":3:1: QML TextEdit: Cannot open: " + testFileUrl("http/notexists.png").toString(); QTest::newRow("local") << testFileUrl("embeddedImagesLocalRelative.qml") << ""; QTest::newRow("remote") << testFileUrl("embeddedImagesRemote.qml") << ""; QTest::newRow("remote-error") << testFileUrl("embeddedImagesRemoteError.qml") << testFileUrl("embeddedImagesRemoteError.qml").toString()+":3:1: QML TextEdit: Error downloading http://127.0.0.1:42332/notexists.png - server replied: Not found"; QTest::newRow("remote") << testFileUrl("embeddedImagesRemoteRelative.qml") << ""; } void tst_qquicktextedit::embeddedImages() { QFETCH(QUrl, qmlfile); QFETCH(QString, error); TestHTTPServer server(42332); server.serveDirectory(testFile("http")); if (!error.isEmpty()) QTest::ignoreMessage(QtWarningMsg, error.toLatin1()); QQmlComponent textComponent(&engine, qmlfile); QQuickTextEdit *textObject = qobject_cast(textComponent.create()); QVERIFY(textObject != 0); QTRY_COMPARE(QQuickTextEditPrivate::get(textObject)->document->resourcesLoading(), 0); QPixmap pm(testFile("http/exists.png")); if (error.isEmpty()) { QCOMPARE(textObject->width(), double(pm.width())); QCOMPARE(textObject->height(), double(pm.height())); } else { QVERIFY(16 != pm.width()); // check test is effective QCOMPARE(textObject->width(), 16.0); // default size of QTextDocument broken image icon QCOMPARE(textObject->height(), 16.0); } delete textObject; } void tst_qquicktextedit::emptytags_QTBUG_22058() { QQuickView canvas(testFileUrl("qtbug-22058.qml")); QVERIFY(canvas.rootObject() != 0); canvas.show(); canvas.requestActivateWindow(); QTest::qWaitForWindowShown(&canvas); QQuickTextEdit *input = qobject_cast(qvariant_cast(canvas.rootObject()->property("inputField"))); QVERIFY(input->hasActiveFocus()); QInputMethodEvent event("", QList()); event.setCommitString("Bold<"); QGuiApplication::sendEvent(input, &event); QCOMPARE(input->text(), QString("Bold<")); event.setCommitString(">"); QGuiApplication::sendEvent(input, &event); QCOMPARE(input->text(), QString("Bold<>")); } QTEST_MAIN(tst_qquicktextedit) #include "tst_qquicktextedit.moc"