2011-04-27 10:05:43 +00:00
|
|
|
/****************************************************************************
|
|
|
|
**
|
2016-01-15 12:36:27 +00:00
|
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
|
|
** Contact: https://www.qt.io/licensing/
|
2011-04-27 10:05:43 +00:00
|
|
|
**
|
|
|
|
** This file is part of the test suite of the Qt Toolkit.
|
|
|
|
**
|
2016-01-15 12:36:27 +00:00
|
|
|
** $QT_BEGIN_LICENSE:GPL-EXCEPT$
|
2012-09-19 12:28:29 +00:00
|
|
|
** Commercial License Usage
|
|
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
|
|
** accordance with the commercial license agreement provided with the
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
2015-01-28 08:44:43 +00:00
|
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
2016-01-15 12:36:27 +00:00
|
|
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
|
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
2012-09-19 12:28:29 +00:00
|
|
|
**
|
2016-01-15 12:36:27 +00:00
|
|
|
** GNU General Public License Usage
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
|
|
** General Public License version 3 as published by the Free Software
|
|
|
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
|
|
|
** included in the packaging of this file. Please review the following
|
|
|
|
** information to ensure the GNU General Public License requirements will
|
|
|
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
2011-04-27 10:05:43 +00:00
|
|
|
**
|
|
|
|
** $QT_END_LICENSE$
|
|
|
|
**
|
|
|
|
****************************************************************************/
|
|
|
|
|
2020-11-26 16:31:50 +00:00
|
|
|
#include <QTest>
|
2011-04-27 10:05:43 +00:00
|
|
|
#include <QtGui>
|
2011-09-16 12:07:05 +00:00
|
|
|
#include <QtWidgets>
|
2011-04-27 10:05:43 +00:00
|
|
|
#include <QtDebug>
|
|
|
|
#include <QPair>
|
|
|
|
#include <QList>
|
|
|
|
#include <QPointer>
|
2020-11-26 16:31:50 +00:00
|
|
|
#include <QSignalSpy>
|
2011-04-27 10:05:43 +00:00
|
|
|
|
2017-10-30 12:45:18 +00:00
|
|
|
#include <QtTest/private/qtesthelpers_p.h>
|
|
|
|
|
2011-10-20 11:17:26 +00:00
|
|
|
#include "../../../../shared/filesystem.h"
|
2011-04-27 10:05:43 +00:00
|
|
|
|
2019-05-17 09:31:00 +00:00
|
|
|
#include <memory>
|
|
|
|
|
|
|
|
Q_DECLARE_METATYPE(QCompleter::CompletionMode)
|
|
|
|
|
2017-10-30 12:45:18 +00:00
|
|
|
using namespace QTestPrivate;
|
2012-12-28 16:35:54 +00:00
|
|
|
|
2011-04-27 10:05:43 +00:00
|
|
|
class CsvCompleter : public QCompleter
|
|
|
|
{
|
|
|
|
Q_OBJECT
|
|
|
|
public:
|
2019-05-17 09:31:00 +00:00
|
|
|
using QCompleter::QCompleter;
|
2011-04-27 10:05:43 +00:00
|
|
|
|
2019-05-17 09:31:00 +00:00
|
|
|
QString pathFromIndex(const QModelIndex& sourceIndex) const override;
|
2011-04-27 10:05:43 +00:00
|
|
|
|
|
|
|
void setCsvCompletion(bool set) { csv = set; }
|
|
|
|
|
|
|
|
protected:
|
2019-05-17 09:31:00 +00:00
|
|
|
QStringList splitPath(const QString &path) const override
|
|
|
|
{
|
2015-10-13 07:46:56 +00:00
|
|
|
return csv ? path.split(QLatin1Char(',')) : QCompleter::splitPath(path);
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
2019-05-17 09:31:00 +00:00
|
|
|
bool csv = true;
|
2011-04-27 10:05:43 +00:00
|
|
|
};
|
|
|
|
|
2019-05-17 09:31:00 +00:00
|
|
|
QString CsvCompleter::pathFromIndex(const QModelIndex &sourceIndex) const
|
2011-04-27 10:05:43 +00:00
|
|
|
{
|
|
|
|
if (!csv)
|
2019-05-17 09:31:00 +00:00
|
|
|
return QCompleter::pathFromIndex(sourceIndex);
|
2011-04-27 10:05:43 +00:00
|
|
|
|
2019-05-17 09:31:00 +00:00
|
|
|
if (!sourceIndex.isValid())
|
2011-04-27 10:05:43 +00:00
|
|
|
return QString();
|
|
|
|
|
2019-05-17 09:31:00 +00:00
|
|
|
QModelIndex idx = sourceIndex;
|
2011-04-27 10:05:43 +00:00
|
|
|
QStringList list;
|
|
|
|
do {
|
|
|
|
QString t = model()->data(idx, completionRole()).toString();
|
|
|
|
list.prepend(t);
|
|
|
|
QModelIndex parent = idx.parent();
|
2019-05-17 09:31:00 +00:00
|
|
|
idx = parent.sibling(parent.row(), sourceIndex.column());
|
2011-04-27 10:05:43 +00:00
|
|
|
} while (idx.isValid());
|
|
|
|
|
2019-05-17 09:31:00 +00:00
|
|
|
return list.count() == 1 ? list.constFirst() : list.join(QLatin1Char(','));
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
class tst_QCompleter : public QObject
|
|
|
|
{
|
|
|
|
Q_OBJECT
|
|
|
|
public:
|
|
|
|
tst_QCompleter();
|
|
|
|
~tst_QCompleter();
|
|
|
|
|
|
|
|
private slots:
|
|
|
|
void getSetCheck();
|
|
|
|
|
|
|
|
void multipleWidgets();
|
|
|
|
void focusIn();
|
|
|
|
|
|
|
|
void csMatchingOnCsSortedModel_data();
|
|
|
|
void csMatchingOnCsSortedModel();
|
|
|
|
void ciMatchingOnCiSortedModel_data();
|
|
|
|
void ciMatchingOnCiSortedModel();
|
|
|
|
|
|
|
|
void ciMatchingOnCsSortedModel_data();
|
|
|
|
void ciMatchingOnCsSortedModel();
|
|
|
|
void csMatchingOnCiSortedModel_data();
|
|
|
|
void csMatchingOnCiSortedModel();
|
|
|
|
|
|
|
|
void fileSystemModel_data();
|
|
|
|
void fileSystemModel();
|
|
|
|
|
|
|
|
void changingModel_data();
|
|
|
|
void changingModel();
|
|
|
|
|
|
|
|
void sortedEngineRowCount_data();
|
|
|
|
void sortedEngineRowCount();
|
|
|
|
void unsortedEngineRowCount_data();
|
|
|
|
void unsortedEngineRowCount();
|
|
|
|
|
|
|
|
void currentRow();
|
|
|
|
void sortedEngineMapFromSource();
|
|
|
|
void unsortedEngineMapFromSource();
|
|
|
|
|
|
|
|
void historySearch();
|
|
|
|
|
|
|
|
void modelDeletion();
|
|
|
|
void setters();
|
|
|
|
|
|
|
|
void dynamicSortOrder();
|
|
|
|
void disabledItems();
|
|
|
|
|
|
|
|
// task-specific tests below me
|
|
|
|
void task178797_activatedOnReturn();
|
|
|
|
void task189564_omitNonSelectableItems();
|
|
|
|
void task246056_setCompletionPrefix();
|
|
|
|
void task250064_lostFocus();
|
|
|
|
|
|
|
|
void task253125_lineEditCompletion_data();
|
|
|
|
void task253125_lineEditCompletion();
|
|
|
|
void task247560_keyboardNavigation();
|
|
|
|
void QTBUG_14292_filesystem();
|
QCompleter: Send activated() signal only once on return key
Due to the complex event forwarding logic between QCompleter,
QComboBox, QLineEdit and QWidgetLineControl, in some cases the
same single user return key press could result in duplicated
activated() signals being emitted by QComboBox. The first one
would be emitted because QLineEdit emitted editingFinished()
as a result of QCompleter::eventFilter() having forwarded the
return key press event to QComboBox. The second one, would
happen right after, as QCompleter::eventFilter() would process
the same event on behalf of its popup.
(We recall that QCompleter is installed as its own popup event
filter. That's also the case for the completer's widget, although
the purpose there is limited to focus-out events).
The current fix consists on skipping the emit as a result of
QLineEdit::editingFinished() if the completer's popup is still
active. For this to be accurate, it helps to test whether the
completer's popup is visible, so we will not be hiding it in
QWidgetLineControl::processKeyEvent() anymore. Indeed, we know
that if the popup is visible, that means that processKeyEvent()
was called after being forwarded by the completer's popup event
filter. Furthermore, the popup will be hidden by its event filter
shortly after it returns from said event forwarding call.
Based on a patch by Alexey Chernov <4ernov@gmail.com>.
Task-number: QTBUG-51858
Task-number: QTBUG-51889
Change-Id: I013f6c3000ae37b5b0ec20eaf5cf7746c9c903e3
Reviewed-by: Richard Moe Gustavsen <richard.gustavsen@qt.io>
2017-08-28 03:04:35 +00:00
|
|
|
void QTBUG_52028_tabAutoCompletes();
|
|
|
|
void QTBUG_51889_activatedSentTwice();
|
2011-04-27 10:05:43 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
void filter(bool assync = false);
|
|
|
|
void testRowCount();
|
|
|
|
enum ModelType {
|
|
|
|
CASE_SENSITIVELY_SORTED_MODEL,
|
|
|
|
CASE_INSENSITIVELY_SORTED_MODEL,
|
|
|
|
HISTORY_MODEL,
|
|
|
|
FILESYSTEM_MODEL
|
|
|
|
};
|
|
|
|
void setSourceModel(ModelType);
|
|
|
|
|
2019-05-17 09:31:00 +00:00
|
|
|
CsvCompleter *completer = nullptr;
|
2011-04-27 10:05:43 +00:00
|
|
|
QTreeWidget *treeWidget;
|
2019-05-17 09:31:00 +00:00
|
|
|
const int completionColumn = 0;
|
|
|
|
const int columnCount = 3;
|
2011-04-27 10:05:43 +00:00
|
|
|
};
|
|
|
|
|
2019-05-17 09:31:00 +00:00
|
|
|
tst_QCompleter::tst_QCompleter() : treeWidget(new QTreeWidget)
|
2011-04-27 10:05:43 +00:00
|
|
|
{
|
2013-04-04 12:09:20 +00:00
|
|
|
treeWidget->move(100, 100);
|
2011-04-27 10:05:43 +00:00
|
|
|
treeWidget->setColumnCount(columnCount);
|
|
|
|
}
|
|
|
|
|
|
|
|
tst_QCompleter::~tst_QCompleter()
|
|
|
|
{
|
|
|
|
delete treeWidget;
|
|
|
|
delete completer;
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::setSourceModel(ModelType type)
|
|
|
|
{
|
|
|
|
QTreeWidgetItem *parent, *child;
|
|
|
|
treeWidget->clear();
|
|
|
|
switch(type) {
|
|
|
|
case CASE_SENSITIVELY_SORTED_MODEL:
|
|
|
|
// Creates a tree model with top level items P0, P1, .., p0, p1,..
|
|
|
|
// Each of these items parents have children (for P0 - c0P0, c1P0,...)
|
|
|
|
for (int i = 0; i < 2; i++) {
|
|
|
|
for (int j = 0; j < 5; j++) {
|
|
|
|
parent = new QTreeWidgetItem(treeWidget);
|
2019-05-17 09:31:00 +00:00
|
|
|
const QString text = QLatin1Char(i == 0 ? 'P' : 'p') + QString::number(j);
|
2011-04-27 10:05:43 +00:00
|
|
|
parent->setText(completionColumn, text);
|
|
|
|
for (int k = 0; k < 5; k++) {
|
|
|
|
child = new QTreeWidgetItem(parent);
|
2019-05-17 09:31:00 +00:00
|
|
|
QString t = QLatin1Char('c') + QString::number(k) + text;
|
2011-04-27 10:05:43 +00:00
|
|
|
child->setText(completionColumn, t);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
completer->setModel(treeWidget->model());
|
|
|
|
completer->setCompletionColumn(completionColumn);
|
|
|
|
break;
|
|
|
|
case CASE_INSENSITIVELY_SORTED_MODEL:
|
|
|
|
case HISTORY_MODEL:
|
|
|
|
// Creates a tree model with top level items P0, p0, P1, p1,...
|
|
|
|
// Each of these items have children c0p0, c1p0,..
|
|
|
|
for (int i = 0; i < 5; i++) {
|
|
|
|
for (int j = 0; j < 2; j++) {
|
|
|
|
parent = new QTreeWidgetItem(treeWidget);
|
2019-05-17 09:31:00 +00:00
|
|
|
const QString text = QLatin1Char(j == 0 ? 'P' : 'p') + QString::number(i);
|
2011-04-27 10:05:43 +00:00
|
|
|
parent->setText(completionColumn, text);
|
|
|
|
for (int k = 0; k < 5; k++) {
|
|
|
|
child = new QTreeWidgetItem(parent);
|
2019-05-17 09:31:00 +00:00
|
|
|
QString t = QLatin1Char('c') + QString::number(k) + text;
|
2011-04-27 10:05:43 +00:00
|
|
|
child->setText(completionColumn, t);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
completer->setModel(treeWidget->model());
|
|
|
|
completer->setCompletionColumn(completionColumn);
|
|
|
|
if (type == CASE_INSENSITIVELY_SORTED_MODEL)
|
|
|
|
break;
|
|
|
|
parent = new QTreeWidgetItem(treeWidget);
|
|
|
|
parent->setText(completionColumn, QLatin1String("p3,c3p3"));
|
|
|
|
parent = new QTreeWidgetItem(treeWidget);
|
|
|
|
parent->setText(completionColumn, QLatin1String("p2,c4p2"));
|
|
|
|
break;
|
|
|
|
case FILESYSTEM_MODEL:
|
|
|
|
completer->setCsvCompletion(false);
|
|
|
|
{
|
2019-05-17 09:31:00 +00:00
|
|
|
auto m = new QFileSystemModel(completer);
|
2011-04-27 10:05:43 +00:00
|
|
|
m->setRootPath("/");
|
|
|
|
completer->setModel(m);
|
|
|
|
}
|
|
|
|
completer->setCompletionColumn(0);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
qDebug() << "Invalid type";
|
2019-05-17 09:31:00 +00:00
|
|
|
break;
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::filter(bool assync)
|
|
|
|
{
|
|
|
|
QFETCH(QString, filterText);
|
2019-05-17 09:31:00 +00:00
|
|
|
QFETCH(const QString, step);
|
2011-04-27 10:05:43 +00:00
|
|
|
QFETCH(QString, completion);
|
|
|
|
QFETCH(QString, completionText);
|
|
|
|
|
|
|
|
if (filterText.compare("FILTERING_OFF", Qt::CaseInsensitive) == 0) {
|
|
|
|
completer->setCompletionMode(QCompleter::UnfilteredPopupCompletion);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-05-17 09:31:00 +00:00
|
|
|
int result = -1;
|
|
|
|
const int attempts = assync ? 10 : 1;
|
|
|
|
for (int times = 0; times < attempts; ++times) {
|
|
|
|
completer->setCompletionPrefix(filterText);
|
|
|
|
|
|
|
|
for (QChar s : step) {
|
|
|
|
int row = completer->currentRow();
|
|
|
|
switch (s.toUpper().toLatin1()) {
|
|
|
|
case 'P':
|
|
|
|
--row;
|
|
|
|
break;
|
|
|
|
case 'N':
|
|
|
|
++row;
|
|
|
|
break;
|
|
|
|
case 'L':
|
|
|
|
row = completer->completionCount() - 1;
|
|
|
|
break;
|
|
|
|
case 'F':
|
|
|
|
row = 0;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
QFAIL(qPrintable(QString(
|
|
|
|
"Problem with 'step' value in test data: %1 (only P, N, L and F are allowed)."
|
|
|
|
).arg(s)));
|
|
|
|
}
|
|
|
|
completer->setCurrentRow(row);
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
|
|
|
|
2019-05-17 09:31:00 +00:00
|
|
|
result = QString::compare(completer->currentCompletion(), completionText,
|
|
|
|
completer->caseSensitivity());
|
|
|
|
if (result == 0)
|
|
|
|
break;
|
|
|
|
if (assync)
|
|
|
|
QTest::qWait(50 * times);
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
2019-05-17 09:31:00 +00:00
|
|
|
|
|
|
|
QCOMPARE(result, 0);
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Testing get/set functions
|
|
|
|
void tst_QCompleter::getSetCheck()
|
|
|
|
{
|
2016-07-12 14:02:53 +00:00
|
|
|
QStandardItemModel standardItemModel(3,3);
|
|
|
|
QCompleter completer(&standardItemModel);
|
2011-04-27 10:05:43 +00:00
|
|
|
|
|
|
|
// QString QCompleter::completionPrefix()
|
|
|
|
// void QCompleter::setCompletionPrefix(QString)
|
|
|
|
completer.setCompletionPrefix(QString("te"));
|
|
|
|
QCOMPARE(completer.completionPrefix(), QString("te"));
|
|
|
|
completer.setCompletionPrefix(QString());
|
|
|
|
QCOMPARE(completer.completionPrefix(), QString());
|
|
|
|
|
|
|
|
// ModelSorting QCompleter::modelSorting()
|
|
|
|
// void QCompleter::setModelSorting(ModelSorting)
|
|
|
|
completer.setModelSorting(QCompleter::CaseSensitivelySortedModel);
|
|
|
|
QCOMPARE(completer.modelSorting(), QCompleter::CaseSensitivelySortedModel);
|
|
|
|
completer.setModelSorting(QCompleter::CaseInsensitivelySortedModel);
|
|
|
|
QCOMPARE(completer.modelSorting(), QCompleter::CaseInsensitivelySortedModel);
|
|
|
|
completer.setModelSorting(QCompleter::UnsortedModel);
|
|
|
|
QCOMPARE(completer.modelSorting(), QCompleter::UnsortedModel);
|
|
|
|
|
|
|
|
// CompletionMode QCompleter::completionMode()
|
|
|
|
// void QCompleter::setCompletionMode(CompletionMode)
|
|
|
|
QCOMPARE(completer.completionMode(), QCompleter::PopupCompletion); // default value
|
|
|
|
completer.setCompletionMode(QCompleter::UnfilteredPopupCompletion);
|
|
|
|
QCOMPARE(completer.completionMode(), QCompleter::UnfilteredPopupCompletion);
|
|
|
|
completer.setCompletionMode(QCompleter::InlineCompletion);
|
|
|
|
QCOMPARE(completer.completionMode(), QCompleter::InlineCompletion);
|
|
|
|
|
|
|
|
// int QCompleter::completionColumn()
|
|
|
|
// void QCompleter::setCompletionColumn(int)
|
|
|
|
completer.setCompletionColumn(2);
|
|
|
|
QCOMPARE(completer.completionColumn(), 2);
|
|
|
|
completer.setCompletionColumn(1);
|
|
|
|
QCOMPARE(completer.completionColumn(), 1);
|
|
|
|
|
|
|
|
// int QCompleter::completionRole()
|
|
|
|
// void QCompleter::setCompletionRole(int)
|
|
|
|
QCOMPARE(completer.completionRole(), static_cast<int>(Qt::EditRole)); // default value
|
|
|
|
completer.setCompletionRole(Qt::DisplayRole);
|
|
|
|
QCOMPARE(completer.completionRole(), static_cast<int>(Qt::DisplayRole));
|
|
|
|
|
|
|
|
// int QCompleter::maxVisibleItems()
|
|
|
|
// void QCompleter::setMaxVisibleItems(int)
|
|
|
|
QCOMPARE(completer.maxVisibleItems(), 7); // default value
|
|
|
|
completer.setMaxVisibleItems(10);
|
|
|
|
QCOMPARE(completer.maxVisibleItems(), 10);
|
|
|
|
QTest::ignoreMessage(QtWarningMsg, "QCompleter::setMaxVisibleItems: "
|
|
|
|
"Invalid max visible items (-2147483648) must be >= 0");
|
|
|
|
completer.setMaxVisibleItems(INT_MIN);
|
|
|
|
QCOMPARE(completer.maxVisibleItems(), 10); // Cannot be set to something negative => old value
|
|
|
|
|
|
|
|
// Qt::CaseSensitivity QCompleter::caseSensitivity()
|
|
|
|
// void QCompleter::setCaseSensitivity(Qt::CaseSensitivity)
|
|
|
|
QCOMPARE(completer.caseSensitivity(), Qt::CaseSensitive); // default value
|
|
|
|
completer.setCaseSensitivity(Qt::CaseInsensitive);
|
|
|
|
QCOMPARE(completer.caseSensitivity(), Qt::CaseInsensitive);
|
|
|
|
|
|
|
|
// bool QCompleter::wrapAround()
|
|
|
|
// void QCompleter::setWrapAround(bool)
|
|
|
|
QCOMPARE(completer.wrapAround(), true); // default value
|
|
|
|
completer.setWrapAround(false);
|
|
|
|
QCOMPARE(completer.wrapAround(), false);
|
2016-07-12 14:02:53 +00:00
|
|
|
|
2017-05-31 21:08:35 +00:00
|
|
|
#if QT_CONFIG(filesystemmodel)
|
2016-07-12 14:02:53 +00:00
|
|
|
// QTBUG-54642, changing from QFileSystemModel to another model should restore role.
|
|
|
|
completer.setCompletionRole(Qt::EditRole);
|
|
|
|
QCOMPARE(completer.completionRole(), static_cast<int>(Qt::EditRole)); // default value
|
|
|
|
QFileSystemModel fileSystemModel;
|
|
|
|
completer.setModel(&fileSystemModel);
|
|
|
|
QCOMPARE(completer.completionRole(), static_cast<int>(QFileSystemModel::FileNameRole));
|
|
|
|
completer.setModel(&standardItemModel);
|
|
|
|
QCOMPARE(completer.completionRole(), static_cast<int>(Qt::EditRole));
|
|
|
|
completer.setCompletionRole(Qt::ToolTipRole);
|
|
|
|
QStandardItemModel standardItemModel2(2, 2); // Do not clobber a custom role when changing models
|
|
|
|
completer.setModel(&standardItemModel2);
|
|
|
|
QCOMPARE(completer.completionRole(), static_cast<int>(Qt::ToolTipRole));
|
2017-05-31 21:08:35 +00:00
|
|
|
#endif // QT_CONFIG(filesystemmodel)
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::csMatchingOnCsSortedModel_data()
|
|
|
|
{
|
|
|
|
delete completer;
|
|
|
|
completer = new CsvCompleter;
|
|
|
|
completer->setModelSorting(QCompleter::CaseSensitivelySortedModel);
|
|
|
|
completer->setCaseSensitivity(Qt::CaseSensitive);
|
|
|
|
setSourceModel(CASE_SENSITIVELY_SORTED_MODEL);
|
|
|
|
|
|
|
|
QTest::addColumn<QString>("filterText");
|
|
|
|
QTest::addColumn<QString>("step");
|
|
|
|
QTest::addColumn<QString>("completion");
|
|
|
|
QTest::addColumn<QString>("completionText");
|
|
|
|
|
2015-10-16 07:38:51 +00:00
|
|
|
#define ROWNAME(name) ((QByteArray(name) + ' ' + QByteArray::number(i)).constData())
|
2012-02-14 03:03:16 +00:00
|
|
|
|
2011-04-27 10:05:43 +00:00
|
|
|
for (int i = 0; i < 2; i++) {
|
|
|
|
if (i == 1)
|
|
|
|
QTest::newRow("FILTERING_OFF") << "FILTERING_OFF" << "" << "" << "";
|
|
|
|
|
|
|
|
// Plain text filter
|
2012-02-14 03:03:16 +00:00
|
|
|
QTest::newRow(ROWNAME("()")) << "" << "" << "P0" << "P0";
|
|
|
|
QTest::newRow(ROWNAME("()F")) << "" << "F" << "P0" << "P0";
|
|
|
|
QTest::newRow(ROWNAME("()L")) << "" << "L" << "p4" << "p4";
|
|
|
|
QTest::newRow(ROWNAME("()N")) << "" << "N" << "P1" << "P1";
|
|
|
|
QTest::newRow(ROWNAME("(P)")) << "P" << "" << "P0" << "P0";
|
|
|
|
QTest::newRow(ROWNAME("(P)F")) << "P" << "" << "P0" << "P0";
|
|
|
|
QTest::newRow(ROWNAME("(P)L")) << "P" << "L" << "P4" << "P4";
|
|
|
|
QTest::newRow(ROWNAME("(p)")) << "p" << "" << "p0" << "p0";
|
|
|
|
QTest::newRow(ROWNAME("(p)N")) << "p" << "N" << "p1" << "p1";
|
|
|
|
QTest::newRow(ROWNAME("(p)NN")) << "p" << "NN" << "p2" << "p2";
|
|
|
|
QTest::newRow(ROWNAME("(p)NNN")) << "p" << "NNN" << "p3" << "p3";
|
|
|
|
QTest::newRow(ROWNAME("(p)NNNN")) << "p" << "NNNN" << "p4" << "p4";
|
|
|
|
QTest::newRow(ROWNAME("(p1)")) << "p1" << "" << "p1" << "p1";
|
|
|
|
QTest::newRow(ROWNAME("(p11)")) << "p11" << "" << "" << "";
|
2011-04-27 10:05:43 +00:00
|
|
|
|
|
|
|
// Tree filter
|
2012-02-14 03:03:16 +00:00
|
|
|
QTest::newRow(ROWNAME("(P0,)")) << "P0," << "" << "c0P0" << "P0,c0P0";
|
|
|
|
QTest::newRow(ROWNAME("(P0,c)")) << "P0,c" << "" << "c0P0" << "P0,c0P0";
|
|
|
|
QTest::newRow(ROWNAME("(P0,c1)")) << "P0,c1" << "" << "c1P0" << "P0,c1P0";
|
|
|
|
QTest::newRow(ROWNAME("(P0,c3P0)")) << "P0,c3P0" << "" << "c3P0" << "P0,c3P0";
|
|
|
|
QTest::newRow(ROWNAME("(P3,c)F")) << "P3,c" << "F" << "c0P3" << "P3,c0P3";
|
|
|
|
QTest::newRow(ROWNAME("(P3,c)L")) << "P3,c" << "L" << "c4P3" << "P3,c4P3";
|
|
|
|
QTest::newRow(ROWNAME("(P3,c)N")) << "P3,c" << "N" << "c1P3" << "P3,c1P3";
|
|
|
|
QTest::newRow(ROWNAME("(P3,c)NN")) << "P3,c" << "NN" << "c2P3" << "P3,c2P3";
|
|
|
|
QTest::newRow(ROWNAME("(P3,,c)")) << "P3,,c" << "" << "" << "";
|
|
|
|
QTest::newRow(ROWNAME("(P3,c0P3,)")) << "P3,c0P3," << "" << "" << "";
|
|
|
|
QTest::newRow(ROWNAME("(P,)")) << "P," << "" << "" << "";
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
2012-02-14 03:03:16 +00:00
|
|
|
#undef ROWNAME
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::csMatchingOnCsSortedModel()
|
|
|
|
{
|
|
|
|
filter();
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::ciMatchingOnCiSortedModel_data()
|
|
|
|
{
|
|
|
|
delete completer;
|
|
|
|
completer = new CsvCompleter;
|
|
|
|
completer->setModelSorting(QCompleter::CaseInsensitivelySortedModel);
|
|
|
|
completer->setCaseSensitivity(Qt::CaseInsensitive);
|
|
|
|
setSourceModel(CASE_INSENSITIVELY_SORTED_MODEL);
|
|
|
|
|
|
|
|
QTest::addColumn<QString>("filterText");
|
|
|
|
QTest::addColumn<QString>("step");
|
|
|
|
QTest::addColumn<QString>("completion");
|
|
|
|
QTest::addColumn<QString>("completionText");
|
|
|
|
|
|
|
|
for (int i = 0; i < 2; i++) {
|
|
|
|
if (i == 1)
|
|
|
|
QTest::newRow("FILTERING_OFF") << "FILTERING_OFF" << "" << "" << "";
|
|
|
|
|
|
|
|
// Plain text filter
|
|
|
|
QTest::newRow("()") << "" << "" << "P0" << "P0";
|
|
|
|
QTest::newRow("()F") << "" << "F" << "P0" << "P0";
|
|
|
|
QTest::newRow("()L") << "" << "L" << "p4" << "p4";
|
|
|
|
QTest::newRow("()N") << "" << "N" << "p0" << "p0";
|
|
|
|
QTest::newRow("(P)") << "P" << "" << "P0" << "P0";
|
|
|
|
QTest::newRow("(P)F") << "P" << "" << "P0" << "P0";
|
|
|
|
QTest::newRow("(P)L") << "P" << "L" << "p4" << "p4";
|
|
|
|
QTest::newRow("(p)") << "p" << "" << "P0" << "P0";
|
|
|
|
QTest::newRow("(p)N") << "p" << "N" << "p0" << "p0";
|
|
|
|
QTest::newRow("(p)NN") << "p" << "NN" << "P1" << "P1";
|
|
|
|
QTest::newRow("(p)NNN") << "p" << "NNN" << "p1" << "p1";
|
|
|
|
QTest::newRow("(p1)") << "p1" << "" << "P1" << "P1";
|
|
|
|
QTest::newRow("(p1)N") << "p1" << "N" << "p1" << "p1";
|
|
|
|
QTest::newRow("(p11)") << "p11" << "" << "" << "";
|
|
|
|
|
|
|
|
//// Tree filter
|
|
|
|
QTest::newRow("(p0,)") << "p0," << "" << "c0P0" << "P0,c0P0";
|
|
|
|
QTest::newRow("(p0,c)") << "p0,c" << "" << "c0P0" << "P0,c0P0";
|
|
|
|
QTest::newRow("(p0,c1)") << "p0,c1" << "" << "c1P0" << "P0,c1P0";
|
|
|
|
QTest::newRow("(p0,c3P0)") << "p0,c3P0" << "" << "c3P0" << "P0,c3P0";
|
|
|
|
QTest::newRow("(p3,c)F") << "p3,c" << "F" << "c0P3" << "P3,c0P3";
|
|
|
|
QTest::newRow("(p3,c)L") << "p3,c" << "L" << "c4P3" << "P3,c4P3";
|
|
|
|
QTest::newRow("(p3,c)N") << "p3,c" << "N" << "c1P3" << "P3,c1P3";
|
|
|
|
QTest::newRow("(p3,c)NN") << "p3,c" << "NN" << "c2P3" << "P3,c2P3";
|
|
|
|
QTest::newRow("(p3,,c)") << "p3,,c" << "" << "" << "";
|
|
|
|
QTest::newRow("(p3,c0P3,)") << "p3,c0P3," << "" << "" << "";
|
|
|
|
QTest::newRow("(p,)") << "p," << "" << "" << "";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::ciMatchingOnCiSortedModel()
|
|
|
|
{
|
|
|
|
filter();
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::ciMatchingOnCsSortedModel_data()
|
|
|
|
{
|
|
|
|
delete completer;
|
|
|
|
completer = new CsvCompleter;
|
|
|
|
completer->setModelSorting(QCompleter::CaseSensitivelySortedModel);
|
|
|
|
setSourceModel(CASE_SENSITIVELY_SORTED_MODEL);
|
|
|
|
completer->setCaseSensitivity(Qt::CaseInsensitive);
|
|
|
|
|
|
|
|
QTest::addColumn<QString>("filterText");
|
|
|
|
QTest::addColumn<QString>("step");
|
|
|
|
QTest::addColumn<QString>("completion");
|
|
|
|
QTest::addColumn<QString>("completionText");
|
|
|
|
|
|
|
|
for (int i = 0; i < 2; i++) {
|
|
|
|
if (i == 1)
|
|
|
|
QTest::newRow("FILTERING_OFF") << "FILTERING_OFF" << "" << "" << "";
|
|
|
|
|
|
|
|
// Plain text filter
|
|
|
|
QTest::newRow("()") << "" << "" << "P0" << "P0";
|
|
|
|
QTest::newRow("()F") << "" << "F" << "P0" << "P0";
|
|
|
|
QTest::newRow("()L") << "" << "L" << "p4" << "p4";
|
|
|
|
QTest::newRow("(P)") << "P" << "" << "P0" << "P0";
|
|
|
|
QTest::newRow("(P)F") << "P" << "" << "P0" << "P0";
|
|
|
|
QTest::newRow("(P)L") << "P" << "L" << "p4" << "p4";
|
|
|
|
QTest::newRow("(p)") << "p" << "" << "P0" << "P0";
|
|
|
|
QTest::newRow("(p)N") << "p" << "N" << "P1" << "P1";
|
|
|
|
QTest::newRow("(p)NN") << "p" << "NN" << "P2" << "P2";
|
|
|
|
QTest::newRow("(p)NNN") << "p" << "NNN" << "P3" << "P3";
|
|
|
|
QTest::newRow("(p1)") << "p1" << "" << "P1" << "P1";
|
|
|
|
QTest::newRow("(p1)N") << "p1" << "N" << "p1" << "p1";
|
|
|
|
QTest::newRow("(p11)") << "p11" << "" << "" << "";
|
|
|
|
|
|
|
|
// Tree filter
|
|
|
|
QTest::newRow("(p0,)") << "p0," << "" << "c0P0" << "P0,c0P0";
|
|
|
|
QTest::newRow("(p0,c)") << "p0,c" << "" << "c0P0" << "P0,c0P0";
|
|
|
|
QTest::newRow("(p0,c1)") << "p0,c1" << "" << "c1P0" << "P0,c1P0";
|
|
|
|
QTest::newRow("(p0,c3P0)") << "p0,c3P0" << "" << "c3P0" << "P0,c3P0";
|
|
|
|
QTest::newRow("(p3,c)F") << "p3,c" << "F" << "c0P3" << "P3,c0P3";
|
|
|
|
QTest::newRow("(p3,c)L") << "p3,c" << "L" << "c4P3" << "P3,c4P3";
|
|
|
|
QTest::newRow("(p3,c)N") << "p3,c" << "N" << "c1P3" << "P3,c1P3";
|
|
|
|
QTest::newRow("(p3,c)NN") << "p3,c" << "NN" << "c2P3" << "P3,c2P3";
|
|
|
|
QTest::newRow("(p3,,c)") << "p3,,c" << "" << "" << "";
|
|
|
|
QTest::newRow("(p3,c0P3,)") << "p3,c0P3," << "" << "" << "";
|
|
|
|
QTest::newRow("(p,)") << "p," << "" << "" << "";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::ciMatchingOnCsSortedModel()
|
|
|
|
{
|
|
|
|
filter();
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::csMatchingOnCiSortedModel_data()
|
|
|
|
{
|
|
|
|
delete completer;
|
|
|
|
completer = new CsvCompleter;
|
|
|
|
completer->setModelSorting(QCompleter::CaseInsensitivelySortedModel);
|
|
|
|
setSourceModel(CASE_INSENSITIVELY_SORTED_MODEL);
|
|
|
|
completer->setCaseSensitivity(Qt::CaseSensitive);
|
|
|
|
|
|
|
|
QTest::addColumn<QString>("filterText");
|
|
|
|
QTest::addColumn<QString>("step");
|
|
|
|
QTest::addColumn<QString>("completion");
|
|
|
|
QTest::addColumn<QString>("completionText");
|
|
|
|
|
|
|
|
for (int i = 0; i < 2; i++) {
|
|
|
|
if (i == 1)
|
|
|
|
QTest::newRow("FILTERING_OFF") << "FILTERING_OFF" << "" << "" << "";
|
|
|
|
|
|
|
|
// Plain text filter
|
|
|
|
QTest::newRow("()") << "" << "" << "P0" << "P0";
|
|
|
|
QTest::newRow("()F") << "" << "F" << "P0" << "P0";
|
|
|
|
QTest::newRow("()L") << "" << "L" << "p4" << "p4";
|
|
|
|
QTest::newRow("()N") << "" << "N" << "p0" << "p0";
|
|
|
|
QTest::newRow("(P)") << "P" << "" << "P0" << "P0";
|
|
|
|
QTest::newRow("(P)F") << "P" << "" << "P0" << "P0";
|
|
|
|
QTest::newRow("(P)L") << "P" << "L" << "P4" << "P4";
|
|
|
|
QTest::newRow("(p)") << "p" << "" << "p0" << "p0";
|
|
|
|
QTest::newRow("(p)N") << "p" << "N" << "p1" << "p1";
|
|
|
|
QTest::newRow("(p)NN") << "p" << "NN" << "p2" << "p2";
|
|
|
|
QTest::newRow("(p)NNN") << "p" << "NNN" << "p3" << "p3";
|
|
|
|
QTest::newRow("(p1)") << "p1" << "" << "p1" << "p1";
|
|
|
|
QTest::newRow("(p11)") << "p11" << "" << "" << "";
|
|
|
|
|
|
|
|
//// Tree filter
|
|
|
|
QTest::newRow("(p0,)") << "p0," << "" << "c0p0" << "p0,c0p0";
|
|
|
|
QTest::newRow("(p0,c)") << "p0,c" << "" << "c0p0" << "p0,c0p0";
|
|
|
|
QTest::newRow("(p0,c1)") << "p0,c1" << "" << "c1p0" << "p0,c1p0";
|
|
|
|
QTest::newRow("(p0,c3P0)") << "p0,c3p0" << "" << "c3p0" << "p0,c3p0";
|
|
|
|
QTest::newRow("(p3,c)F") << "p3,c" << "F" << "c0p3" << "p3,c0p3";
|
|
|
|
QTest::newRow("(p3,c)L") << "p3,c" << "L" << "c4p3" << "p3,c4p3";
|
|
|
|
QTest::newRow("(p3,c)N") << "p3,c" << "N" << "c1p3" << "p3,c1p3";
|
|
|
|
QTest::newRow("(p3,c)NN") << "p3,c" << "NN" << "c2p3" << "p3,c2p3";
|
|
|
|
QTest::newRow("(p3,,c)") << "p3,,c" << "" << "" << "";
|
|
|
|
QTest::newRow("(p3,c0P3,)") << "p3,c0P3," << "" << "" << "";
|
|
|
|
QTest::newRow("(p,)") << "p," << "" << "" << "";
|
|
|
|
|
|
|
|
QTest::newRow("FILTERING_OFF") << "FILTERING_OFF" << "" << "" << "";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::csMatchingOnCiSortedModel()
|
|
|
|
{
|
|
|
|
filter();
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::fileSystemModel_data()
|
|
|
|
{
|
|
|
|
delete completer;
|
|
|
|
completer = new CsvCompleter;
|
|
|
|
completer->setModelSorting(QCompleter::CaseSensitivelySortedModel);
|
|
|
|
setSourceModel(FILESYSTEM_MODEL);
|
|
|
|
completer->setCaseSensitivity(Qt::CaseInsensitive);
|
|
|
|
|
|
|
|
QTest::addColumn<QString>("filterText");
|
|
|
|
QTest::addColumn<QString>("step");
|
|
|
|
QTest::addColumn<QString>("completion");
|
|
|
|
QTest::addColumn<QString>("completionText");
|
|
|
|
|
|
|
|
// NOTE: Add tests carefully, ensurely the paths exist on all systems
|
|
|
|
// Output is the sourceText; currentCompletionText()
|
|
|
|
|
|
|
|
for (int i = 0; i < 2; i++) {
|
|
|
|
if (i == 1)
|
|
|
|
QTest::newRow("FILTERING_OFF") << "FILTERING_OFF" << "" << "" << "";
|
|
|
|
|
2016-03-08 14:51:15 +00:00
|
|
|
#if defined(Q_OS_WIN)
|
2011-04-27 10:05:43 +00:00
|
|
|
QTest::newRow("()") << "C" << "" << "C:" << "C:";
|
|
|
|
QTest::newRow("()") << "C:\\Program" << "" << "Program Files" << "C:\\Program Files";
|
|
|
|
#elif defined (Q_OS_MAC)
|
|
|
|
QTest::newRow("()") << "" << "" << "/" << "/";
|
|
|
|
QTest::newRow("(/a)") << "/a" << "" << "Applications" << "/Applications";
|
|
|
|
// QTest::newRow("(/d)") << "/d" << "" << "Developer" << "/Developer";
|
2015-01-14 11:44:29 +00:00
|
|
|
#elif defined(Q_OS_ANDROID)
|
|
|
|
QTest::newRow("()") << "" << "" << "/" << "/";
|
|
|
|
QTest::newRow("(/et)") << "/et" << "" << "etc" << "/etc";
|
2011-04-27 10:05:43 +00:00
|
|
|
#else
|
|
|
|
QTest::newRow("()") << "" << "" << "/" << "/";
|
2017-10-01 01:19:36 +00:00
|
|
|
#if !defined(Q_OS_AIX) && !defined(Q_OS_HPUX) && !defined(Q_OS_QNX)
|
2011-04-27 10:05:43 +00:00
|
|
|
QTest::newRow("(/h)") << "/h" << "" << "home" << "/home";
|
|
|
|
#endif
|
|
|
|
QTest::newRow("(/et)") << "/et" << "" << "etc" << "/etc";
|
|
|
|
QTest::newRow("(/etc/passw)") << "/etc/passw" << "" << "passwd" << "/etc/passwd";
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::fileSystemModel()
|
|
|
|
{
|
|
|
|
//QFileSystemModel is assync.
|
|
|
|
filter(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void tst_QCompleter::changingModel_data()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::changingModel()
|
|
|
|
{
|
|
|
|
for (int i = 0; i < 2; i++) {
|
|
|
|
delete completer;
|
|
|
|
completer = new CsvCompleter;
|
|
|
|
completer->setModelSorting(QCompleter::CaseSensitivelySortedModel);
|
|
|
|
completer->setCaseSensitivity(Qt::CaseSensitive);
|
|
|
|
setSourceModel(CASE_SENSITIVELY_SORTED_MODEL);
|
|
|
|
|
|
|
|
if (i == 1) {
|
|
|
|
completer->setCompletionMode(QCompleter::UnfilteredPopupCompletion);
|
|
|
|
}
|
|
|
|
|
|
|
|
completer->setCompletionPrefix("p");
|
|
|
|
completer->setCurrentRow(completer->completionCount() - 1);
|
|
|
|
QCOMPARE(completer->currentCompletion(), QString("p4"));
|
|
|
|
|
|
|
|
// Test addition of data
|
|
|
|
QTreeWidgetItem p5item;
|
|
|
|
p5item.setText(completionColumn, "p5");
|
|
|
|
treeWidget->addTopLevelItem(&p5item);
|
|
|
|
completer->setCompletionPrefix("p5");
|
|
|
|
QCOMPARE(completer->currentCompletion(), QString("p5"));
|
|
|
|
|
|
|
|
// Test removal of data
|
|
|
|
int p5index = treeWidget->indexOfTopLevelItem(&p5item);
|
|
|
|
treeWidget->takeTopLevelItem(p5index);
|
|
|
|
QCOMPARE(completer->currentCompletion(), QString(""));
|
|
|
|
|
|
|
|
// Test clear
|
|
|
|
treeWidget->clear();
|
|
|
|
QCOMPARE(completer->currentIndex(), QModelIndex());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::testRowCount()
|
|
|
|
{
|
|
|
|
QFETCH(QString, filterText);
|
|
|
|
QFETCH(bool, hasChildren);
|
|
|
|
QFETCH(int, rowCount);
|
|
|
|
QFETCH(int, completionCount);
|
|
|
|
|
|
|
|
if (filterText.compare("FILTERING_OFF", Qt::CaseInsensitive) == 0) {
|
|
|
|
completer->setCompletionMode(QCompleter::UnfilteredPopupCompletion);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
completer->setCompletionPrefix(filterText);
|
|
|
|
const QAbstractItemModel *completionModel = completer->completionModel();
|
|
|
|
QCOMPARE(completionModel->rowCount(), rowCount);
|
|
|
|
QCOMPARE(completionCount, completionCount);
|
|
|
|
QCOMPARE(completionModel->hasChildren(), hasChildren);
|
|
|
|
QCOMPARE(completionModel->columnCount(), columnCount);
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::sortedEngineRowCount_data()
|
|
|
|
{
|
|
|
|
delete completer;
|
|
|
|
completer = new CsvCompleter;
|
|
|
|
completer->setModelSorting(QCompleter::CaseInsensitivelySortedModel);
|
|
|
|
completer->setCaseSensitivity(Qt::CaseInsensitive);
|
|
|
|
setSourceModel(CASE_INSENSITIVELY_SORTED_MODEL);
|
|
|
|
|
|
|
|
QTest::addColumn<QString>("filterText");
|
|
|
|
QTest::addColumn<bool>("hasChildren");
|
|
|
|
QTest::addColumn<int>("rowCount");
|
|
|
|
QTest::addColumn<int>("completionCount");
|
|
|
|
|
|
|
|
QTest::newRow("whatever") << "whatever" << false << 0 << 0;
|
|
|
|
QTest::newRow("p") << "p" << true << 10 << 10;
|
|
|
|
QTest::newRow("p1") << "p1" << true << 2 << 2;
|
|
|
|
QTest::newRow("P1,") << "P1," << true << 5 << 5;
|
|
|
|
QTest::newRow("P1,c") << "P1,c" << true << 5 << 5;
|
|
|
|
QTest::newRow("P1,cc") << "P1,cc" << false << 0 << 0;
|
|
|
|
|
|
|
|
QTest::newRow("FILTERING_OFF") << "FILTERING_OFF" << false << 0 << 0;
|
|
|
|
|
|
|
|
QTest::newRow("whatever(filter off)") << "whatever" << true << 10 << 0;
|
|
|
|
QTest::newRow("p1(filter off)") << "p1" << true << 10 << 2;
|
|
|
|
QTest::newRow("p1,(filter off)") << "p1," << true << 5 << 5;
|
|
|
|
QTest::newRow("p1,c(filter off)") << "p1,c" << true << 5 << 5;
|
|
|
|
QTest::newRow("P1,cc(filter off)") << "P1,cc" << true << 5 << 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::sortedEngineRowCount()
|
|
|
|
{
|
|
|
|
testRowCount();
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::unsortedEngineRowCount_data()
|
|
|
|
{
|
|
|
|
delete completer;
|
|
|
|
completer = new CsvCompleter;
|
|
|
|
completer->setModelSorting(QCompleter::CaseInsensitivelySortedModel);
|
|
|
|
completer->setCaseSensitivity(Qt::CaseSensitive);
|
|
|
|
setSourceModel(CASE_INSENSITIVELY_SORTED_MODEL);
|
|
|
|
|
|
|
|
QTest::addColumn<QString>("filterText");
|
|
|
|
QTest::addColumn<bool>("hasChildren");
|
|
|
|
QTest::addColumn<int>("rowCount");
|
|
|
|
QTest::addColumn<int>("completionCount");
|
|
|
|
|
|
|
|
QTest::newRow("whatever") << "whatever" << false << 0 << 0;
|
|
|
|
QTest::newRow("p") << "p" << true << 5 << 5;
|
|
|
|
QTest::newRow("p1") << "p1" << true << 1 << 1;
|
|
|
|
QTest::newRow("P1,") << "P1," << true << 5 << 5;
|
|
|
|
QTest::newRow("P1,c") << "P1,c" << true << 5 << 5;
|
|
|
|
QTest::newRow("P1,cc") << "P1,cc" << false << 0 << 0;
|
|
|
|
|
|
|
|
QTest::newRow("FILTERING_OFF") << "FILTERING_OFF" << false << 0 << 0;
|
|
|
|
|
|
|
|
QTest::newRow("whatever(filter off)") << "whatever" << true << 10 << 0;
|
|
|
|
QTest::newRow("p1(filter off)") << "p1" << true << 10 << 1;
|
|
|
|
QTest::newRow("p1,(filter off)") << "p1," << true << 5 << 5;
|
|
|
|
QTest::newRow("p1,c(filter off)") << "p1,c" << true << 5 << 5;
|
|
|
|
QTest::newRow("P1,cc(filter off)") << "P1,cc" << true << 5 << 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::unsortedEngineRowCount()
|
|
|
|
{
|
|
|
|
testRowCount();
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::currentRow()
|
|
|
|
{
|
|
|
|
delete completer;
|
|
|
|
completer = new CsvCompleter;
|
|
|
|
completer->setModelSorting(QCompleter::CaseInsensitivelySortedModel);
|
|
|
|
completer->setCaseSensitivity(Qt::CaseInsensitive);
|
|
|
|
setSourceModel(CASE_INSENSITIVELY_SORTED_MODEL);
|
|
|
|
|
|
|
|
// blank text
|
|
|
|
completer->setCompletionPrefix("");
|
|
|
|
QCOMPARE(completer->currentRow(), 0);
|
|
|
|
QVERIFY(completer->setCurrentRow(4));
|
|
|
|
QCOMPARE(completer->currentRow(), 4);
|
|
|
|
QVERIFY(!completer->setCurrentRow(13));
|
|
|
|
QVERIFY(completer->setCurrentRow(4));
|
|
|
|
|
|
|
|
// some text
|
|
|
|
completer->setCompletionPrefix("p1");
|
|
|
|
QCOMPARE(completer->currentRow(), 0);
|
|
|
|
QVERIFY(completer->setCurrentRow(1));
|
|
|
|
QCOMPARE(completer->currentRow(), 1);
|
|
|
|
QVERIFY(!completer->setCurrentRow(2));
|
|
|
|
QCOMPARE(completer->currentRow(), 1);
|
|
|
|
|
|
|
|
// invalid text
|
|
|
|
completer->setCompletionPrefix("well");
|
|
|
|
QCOMPARE(completer->currentRow(), -1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::sortedEngineMapFromSource()
|
|
|
|
{
|
|
|
|
delete completer;
|
|
|
|
completer = new CsvCompleter;
|
|
|
|
completer->setModelSorting(QCompleter::CaseInsensitivelySortedModel);
|
|
|
|
completer->setCaseSensitivity(Qt::CaseInsensitive);
|
|
|
|
setSourceModel(CASE_INSENSITIVELY_SORTED_MODEL);
|
|
|
|
|
|
|
|
QModelIndex si1, si2, pi;
|
|
|
|
QAbstractItemModel *sourceModel = completer->model();
|
2019-05-17 09:31:00 +00:00
|
|
|
auto completionModel = qobject_cast<const QAbstractProxyModel *>(completer->completionModel());
|
2011-04-27 10:05:43 +00:00
|
|
|
|
|
|
|
// Fitering ON
|
|
|
|
// empty
|
|
|
|
si1 = sourceModel->index(4, completionColumn); // "P2"
|
|
|
|
si2 = sourceModel->index(2, 0, si1); // "P2,c0P2"
|
|
|
|
pi = completionModel->mapFromSource(si1);
|
|
|
|
QCOMPARE(completionModel->data(pi).toString(), QLatin1String("P2"));
|
|
|
|
pi = completionModel->mapFromSource(si2);
|
|
|
|
QCOMPARE(pi.isValid(), false);
|
|
|
|
|
|
|
|
// some text
|
|
|
|
completer->setCompletionPrefix("p");
|
|
|
|
pi = completionModel->mapFromSource(si1);
|
|
|
|
QCOMPARE(completionModel->data(pi).toString(), QLatin1String("P2"));
|
|
|
|
pi = completionModel->mapFromSource(si2);
|
|
|
|
QCOMPARE(pi.isValid(), false);
|
|
|
|
|
|
|
|
// more text
|
|
|
|
completer->setCompletionPrefix("p2");
|
|
|
|
pi = completionModel->mapFromSource(si1);
|
|
|
|
QCOMPARE(completionModel->data(pi).toString(), QLatin1String("P2"));
|
|
|
|
pi = completionModel->mapFromSource(si2);
|
|
|
|
QCOMPARE(pi.isValid(), false);
|
|
|
|
|
|
|
|
// invalid text
|
|
|
|
completer->setCompletionPrefix("whatever");
|
|
|
|
pi = completionModel->mapFromSource(si1);
|
|
|
|
QVERIFY(!pi.isValid());
|
|
|
|
|
|
|
|
// Fitering OFF
|
|
|
|
completer->setCompletionMode(QCompleter::UnfilteredPopupCompletion);
|
|
|
|
// empty
|
|
|
|
si1 = sourceModel->index(4, completionColumn); // "P2"
|
|
|
|
si2 = sourceModel->index(2, 0, si1); // "P2,c0P2"
|
|
|
|
pi = completionModel->mapFromSource(si1);
|
|
|
|
QCOMPARE(completionModel->data(pi).toString(), QLatin1String("P2"));
|
|
|
|
pi = completionModel->mapFromSource(si2);
|
|
|
|
QCOMPARE(pi.isValid(), false);
|
|
|
|
|
|
|
|
// some text
|
|
|
|
completer->setCompletionPrefix("p");
|
|
|
|
pi = completionModel->mapFromSource(si1);
|
|
|
|
QCOMPARE(completionModel->data(pi).toString(), QLatin1String("P2"));
|
|
|
|
pi = completionModel->mapFromSource(si2);
|
|
|
|
QCOMPARE(pi.isValid(), false);
|
|
|
|
|
|
|
|
// more text
|
|
|
|
completer->setCompletionPrefix("p2");
|
|
|
|
pi = completionModel->mapFromSource(si1);
|
|
|
|
QCOMPARE(completionModel->data(pi).toString(), QLatin1String("P2"));
|
|
|
|
pi = completionModel->mapFromSource(si2);
|
|
|
|
QCOMPARE(pi.isValid(), false);
|
|
|
|
|
|
|
|
// invalid text
|
|
|
|
completer->setCompletionPrefix("whatever");
|
|
|
|
pi = completionModel->mapFromSource(si1);
|
|
|
|
QCOMPARE(completionModel->data(pi).toString(), QLatin1String("P2"));
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::unsortedEngineMapFromSource()
|
|
|
|
{
|
|
|
|
delete completer;
|
|
|
|
completer = new CsvCompleter;
|
|
|
|
completer->setCaseSensitivity(Qt::CaseInsensitive);
|
|
|
|
setSourceModel(HISTORY_MODEL); // case insensitively sorted model
|
|
|
|
completer->setModelSorting(QCompleter::UnsortedModel);
|
|
|
|
|
|
|
|
QModelIndex si, si2, si3, pi;
|
|
|
|
QAbstractItemModel *sourceModel = completer->model();
|
2019-05-17 09:31:00 +00:00
|
|
|
auto completionModel = qobject_cast<const QAbstractProxyModel *>(completer->completionModel());
|
2011-04-27 10:05:43 +00:00
|
|
|
|
|
|
|
si = sourceModel->index(6, completionColumn); // "P3"
|
|
|
|
QCOMPARE(si.data().toString(), QLatin1String("P3"));
|
|
|
|
si2 = sourceModel->index(3, completionColumn, sourceModel->index(0, completionColumn)); // "P0,c3P0"
|
|
|
|
QCOMPARE(si2.data().toString(), QLatin1String("c3P0"));
|
|
|
|
si3 = sourceModel->index(10, completionColumn); // "p3,c3p3" (history)
|
|
|
|
QCOMPARE(si3.data().toString(), QLatin1String("p3,c3p3"));
|
|
|
|
|
|
|
|
// FILTERING ON
|
|
|
|
// empty
|
|
|
|
pi = completionModel->mapFromSource(si);
|
|
|
|
QCOMPARE(completionModel->data(pi).toString(), QLatin1String("P3"));
|
|
|
|
pi = completionModel->mapFromSource(si2);
|
|
|
|
QCOMPARE(pi.isValid(), false);
|
|
|
|
pi = completionModel->mapFromSource(si3);
|
|
|
|
QCOMPARE(completionModel->data(pi).toString(), QLatin1String("p3,c3p3"));
|
|
|
|
|
|
|
|
// some text
|
|
|
|
completer->setCompletionPrefix("P");
|
|
|
|
pi = completionModel->mapFromSource(si);
|
|
|
|
QCOMPARE(completionModel->data(pi).toString(), QLatin1String("P3"));
|
|
|
|
pi = completionModel->mapFromSource(si2);
|
|
|
|
QCOMPARE(pi.isValid(), false);
|
|
|
|
pi = completionModel->mapFromSource(si3);
|
|
|
|
QCOMPARE(completionModel->data(pi).toString(), QLatin1String("p3,c3p3"));
|
|
|
|
|
|
|
|
// invalid text
|
|
|
|
completer->setCompletionPrefix("whatever");
|
|
|
|
pi = completionModel->mapFromSource(si);
|
|
|
|
QVERIFY(!pi.isValid());
|
|
|
|
pi = completionModel->mapFromSource(si2);
|
|
|
|
QVERIFY(!pi.isValid());
|
|
|
|
|
|
|
|
// tree matching
|
|
|
|
completer->setCompletionPrefix("P0,c");
|
|
|
|
pi = completionModel->mapFromSource(si);
|
|
|
|
QVERIFY(!pi.isValid());
|
|
|
|
pi = completionModel->mapFromSource(si2);
|
|
|
|
QCOMPARE(completionModel->data(pi).toString(), QLatin1String("c3P0"));
|
|
|
|
pi = completionModel->mapFromSource(si3);
|
|
|
|
QCOMPARE(pi.isValid(), false);
|
|
|
|
|
|
|
|
// more tree matching
|
|
|
|
completer->setCompletionPrefix("p3,");
|
|
|
|
pi = completionModel->mapFromSource(si2);
|
|
|
|
QVERIFY(!pi.isValid());
|
|
|
|
pi = completionModel->mapFromSource(si3);
|
|
|
|
QCOMPARE(completionModel->data(pi).toString(), QLatin1String("p3,c3p3"));
|
|
|
|
|
|
|
|
// FILTERING OFF
|
|
|
|
// empty
|
|
|
|
completer->setCompletionPrefix("");
|
|
|
|
completer->setCompletionMode(QCompleter::UnfilteredPopupCompletion);
|
|
|
|
pi = completionModel->mapFromSource(si);
|
|
|
|
QCOMPARE(completionModel->data(pi).toString(), QLatin1String("P3"));
|
|
|
|
|
|
|
|
// some text
|
|
|
|
completer->setCompletionPrefix("P");
|
|
|
|
pi = completionModel->mapFromSource(si);
|
|
|
|
QCOMPARE(completionModel->data(pi).toString(), QLatin1String("P3"));
|
|
|
|
|
|
|
|
// more text
|
|
|
|
completer->setCompletionPrefix("P3");
|
|
|
|
pi = completionModel->mapFromSource(si);
|
|
|
|
QCOMPARE(completionModel->data(pi).toString(), QLatin1String("P3"));
|
|
|
|
|
|
|
|
// invalid text
|
|
|
|
completer->setCompletionPrefix("whatever");
|
|
|
|
pi = completionModel->mapFromSource(si);
|
|
|
|
QCOMPARE(completionModel->data(pi).toString(), QLatin1String("P3"));
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::historySearch()
|
|
|
|
{
|
|
|
|
delete completer;
|
|
|
|
completer = new CsvCompleter;
|
|
|
|
completer->setModelSorting(QCompleter::CaseInsensitivelySortedModel);
|
|
|
|
completer->setCaseSensitivity(Qt::CaseSensitive);
|
|
|
|
setSourceModel(HISTORY_MODEL);
|
|
|
|
|
2019-05-17 09:31:00 +00:00
|
|
|
auto completionModel = qobject_cast<const QAbstractProxyModel *>(completer->completionModel());
|
2011-04-27 10:05:43 +00:00
|
|
|
|
|
|
|
// "p3,c3p3" and "p2,c4p2" are added in the tree root
|
|
|
|
|
|
|
|
// FILTERING ON
|
|
|
|
// empty
|
|
|
|
completer->setCurrentRow(10);
|
|
|
|
QCOMPARE(completer->currentCompletion(), QLatin1String("p3,c3p3"));
|
|
|
|
|
|
|
|
// more text
|
|
|
|
completer->setCompletionPrefix("p2");
|
|
|
|
completer->setCurrentRow(1);
|
|
|
|
QCOMPARE(completer->currentCompletion(), QLatin1String("p2,c4p2"));
|
|
|
|
|
|
|
|
// comma separated text
|
|
|
|
completer->setCompletionPrefix("p2,c4");
|
|
|
|
completer->setCurrentRow(1);
|
|
|
|
QCOMPARE(completionModel->rowCount(), 2);
|
|
|
|
QCOMPARE(completer->currentCompletion(), QLatin1String("p2,c4p2"));
|
|
|
|
|
|
|
|
// invalid text
|
|
|
|
completer->setCompletionPrefix("whatever");
|
|
|
|
QCOMPARE(completer->currentCompletion(), QString());
|
|
|
|
|
|
|
|
// FILTERING OFF
|
|
|
|
completer->setCompletionPrefix("");
|
|
|
|
completer->setCompletionMode(QCompleter::UnfilteredPopupCompletion);
|
|
|
|
completer->setCurrentRow(10);
|
|
|
|
QCOMPARE(completer->currentCompletion(), QLatin1String("p3,c3p3"));
|
|
|
|
|
|
|
|
// more text
|
|
|
|
completer->setCompletionPrefix("p2");
|
|
|
|
completer->setCurrentRow(1);
|
|
|
|
QCOMPARE(completer->currentCompletion(), QLatin1String("p2,c4p2"));
|
|
|
|
|
|
|
|
// comma separated text
|
|
|
|
completer->setCompletionPrefix("p2,c4");
|
|
|
|
QCOMPARE(completionModel->rowCount(), 5);
|
|
|
|
|
|
|
|
// invalid text
|
|
|
|
completer->setCompletionPrefix("whatever");
|
|
|
|
QCOMPARE(completer->currentCompletion(), QString());
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::setters()
|
|
|
|
{
|
|
|
|
delete completer;
|
|
|
|
completer = new CsvCompleter;
|
2019-05-17 09:31:00 +00:00
|
|
|
QVERIFY(completer->popup() != nullptr);
|
2019-08-16 09:30:58 +00:00
|
|
|
QPointer<QStandardItemModel> itemModel(new QStandardItemModel(1, 0, completer));
|
2011-04-27 10:05:43 +00:00
|
|
|
QAbstractItemModel *oldModel = completer->model();
|
2019-08-16 09:30:58 +00:00
|
|
|
completer->setModel(itemModel.data());
|
2011-04-27 10:05:43 +00:00
|
|
|
QVERIFY(completer->popup()->model() != oldModel);
|
2015-07-30 13:16:36 +00:00
|
|
|
QCOMPARE(completer->popup()->model(), completer->completionModel());
|
2011-04-27 10:05:43 +00:00
|
|
|
completer->setPopup(new QListView);
|
2015-07-30 13:16:36 +00:00
|
|
|
QCOMPARE(completer->popup()->model(), completer->completionModel());
|
2011-04-27 10:05:43 +00:00
|
|
|
completer->setModel(new QStringListModel(completer));
|
2019-08-16 09:30:58 +00:00
|
|
|
QVERIFY(itemModel.isNull()); // must have been deleted
|
2011-04-27 10:05:43 +00:00
|
|
|
|
2019-05-17 09:31:00 +00:00
|
|
|
completer->setModel(nullptr);
|
|
|
|
completer->setWidget(nullptr);
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::modelDeletion()
|
|
|
|
{
|
|
|
|
delete completer;
|
|
|
|
completer = new CsvCompleter;
|
2019-05-17 09:31:00 +00:00
|
|
|
const QStringList list = {"item1", "item2", "item3"};
|
|
|
|
std::unique_ptr<QStringListModel> listModel(new QStringListModel(list));
|
2011-04-27 10:05:43 +00:00
|
|
|
completer->setCompletionPrefix("i");
|
2019-05-17 09:31:00 +00:00
|
|
|
completer->setModel(listModel.get());
|
2015-07-30 13:16:36 +00:00
|
|
|
QCOMPARE(completer->completionCount(), 3);
|
2019-05-17 09:31:00 +00:00
|
|
|
std::unique_ptr<QListView> view(new QListView);
|
|
|
|
view->setWindowTitle(QLatin1String(QTest::currentTestFunction()));
|
2011-04-27 10:05:43 +00:00
|
|
|
view->setModel(completer->completionModel());
|
2019-05-17 09:31:00 +00:00
|
|
|
listModel.reset();
|
2013-04-04 12:09:20 +00:00
|
|
|
view->move(200, 200);
|
2011-04-27 10:05:43 +00:00
|
|
|
view->show();
|
2019-05-17 09:31:00 +00:00
|
|
|
QCoreApplication::processEvents();
|
2013-04-04 12:09:20 +00:00
|
|
|
view.reset();
|
2015-07-30 13:16:36 +00:00
|
|
|
QCOMPARE(completer->completionCount(), 0);
|
|
|
|
QCOMPARE(completer->currentRow(), -1);
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::multipleWidgets()
|
|
|
|
{
|
2019-10-11 13:21:54 +00:00
|
|
|
if (QGuiApplication::platformName().startsWith(QLatin1String("wayland"), Qt::CaseInsensitive))
|
|
|
|
QSKIP("Wayland: This fails. Figure out why.");
|
|
|
|
|
2011-04-27 10:05:43 +00:00
|
|
|
QStringList list;
|
|
|
|
list << "item1" << "item2" << "item2";
|
|
|
|
QCompleter completer(list);
|
|
|
|
completer.setCompletionMode(QCompleter::InlineCompletion);
|
|
|
|
|
|
|
|
QWidget window;
|
2019-05-17 09:31:00 +00:00
|
|
|
window.setWindowTitle(QLatin1String(QTest::currentTestFunction()));
|
2013-04-04 12:09:20 +00:00
|
|
|
window.move(200, 200);
|
2011-04-27 10:05:43 +00:00
|
|
|
window.show();
|
|
|
|
QApplication::setActiveWindow(&window);
|
2012-07-24 12:29:01 +00:00
|
|
|
QVERIFY(QTest::qWaitForWindowActive(&window));
|
2011-04-27 10:05:43 +00:00
|
|
|
|
|
|
|
QFocusEvent focusIn(QEvent::FocusIn);
|
|
|
|
QFocusEvent focusOut(QEvent::FocusOut);
|
|
|
|
|
2019-05-17 09:31:00 +00:00
|
|
|
auto comboBox = new QComboBox(&window);
|
2011-04-27 10:05:43 +00:00
|
|
|
comboBox->setEditable(true);
|
|
|
|
comboBox->setCompleter(&completer);
|
|
|
|
comboBox->setFocus();
|
|
|
|
comboBox->show();
|
|
|
|
window.activateWindow();
|
|
|
|
QApplication::setActiveWindow(&window);
|
2018-04-04 06:50:22 +00:00
|
|
|
QVERIFY(QTest::qWaitForWindowActive(&window));
|
|
|
|
QCOMPARE(QApplication::focusWidget(), comboBox);
|
2011-04-27 10:05:43 +00:00
|
|
|
comboBox->lineEdit()->setText("it");
|
|
|
|
QCOMPARE(comboBox->currentText(), QString("it")); // should not complete with setText
|
|
|
|
QTest::keyPress(comboBox, 'e');
|
|
|
|
QCOMPARE(comboBox->currentText(), QString("item1"));
|
|
|
|
comboBox->clearEditText();
|
|
|
|
QCOMPARE(comboBox->currentText(), QString("")); // combo box text must not change!
|
|
|
|
|
2019-05-17 09:31:00 +00:00
|
|
|
auto lineEdit = new QLineEdit(&window);
|
2011-04-27 10:05:43 +00:00
|
|
|
lineEdit->setCompleter(&completer);
|
|
|
|
lineEdit->show();
|
|
|
|
lineEdit->setFocus();
|
2015-07-30 13:16:36 +00:00
|
|
|
QTRY_COMPARE(QApplication::focusWidget(), lineEdit);
|
2011-04-27 10:05:43 +00:00
|
|
|
lineEdit->setText("it");
|
|
|
|
QCOMPARE(lineEdit->text(), QString("it")); // should not completer with setText
|
|
|
|
QCOMPARE(comboBox->currentText(), QString("")); // combo box text must not change!
|
|
|
|
QTest::keyPress(lineEdit, 'e');
|
|
|
|
QCOMPARE(lineEdit->text(), QString("item1"));
|
|
|
|
QCOMPARE(comboBox->currentText(), QString("")); // combo box text must not change!
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::focusIn()
|
|
|
|
{
|
2019-10-11 13:21:54 +00:00
|
|
|
if (QGuiApplication::platformName().startsWith(QLatin1String("wayland"), Qt::CaseInsensitive))
|
|
|
|
QSKIP("Wayland: This fails. Figure out why.");
|
|
|
|
|
2019-05-17 09:31:00 +00:00
|
|
|
QCompleter completer({"item1", "item2", "item2"});
|
2011-04-27 10:05:43 +00:00
|
|
|
|
|
|
|
QWidget window;
|
2019-05-17 09:31:00 +00:00
|
|
|
window.setWindowTitle(QLatin1String(QTest::currentTestFunction()));
|
2013-04-04 12:09:20 +00:00
|
|
|
window.move(200, 200);
|
2011-04-27 10:05:43 +00:00
|
|
|
window.show();
|
|
|
|
window.activateWindow();
|
|
|
|
QApplication::setActiveWindow(&window);
|
2013-04-04 12:09:20 +00:00
|
|
|
QVERIFY(QTest::qWaitForWindowActive(&window));
|
2011-04-27 10:05:43 +00:00
|
|
|
|
2019-05-17 09:31:00 +00:00
|
|
|
auto comboBox = new QComboBox(&window);
|
2011-04-27 10:05:43 +00:00
|
|
|
comboBox->setEditable(true);
|
|
|
|
comboBox->setCompleter(&completer);
|
|
|
|
comboBox->show();
|
|
|
|
comboBox->lineEdit()->setText("it");
|
|
|
|
|
2019-05-17 09:31:00 +00:00
|
|
|
auto lineEdit = new QLineEdit(&window);
|
2011-04-27 10:05:43 +00:00
|
|
|
lineEdit->setCompleter(&completer);
|
|
|
|
lineEdit->setText("it");
|
|
|
|
lineEdit->show();
|
|
|
|
|
2019-05-17 09:31:00 +00:00
|
|
|
auto lineEdit2 = new QLineEdit(&window); // has no completer!
|
2011-04-27 10:05:43 +00:00
|
|
|
lineEdit2->show();
|
|
|
|
|
|
|
|
comboBox->setFocus();
|
2015-07-30 13:16:36 +00:00
|
|
|
QTRY_COMPARE(completer.widget(), comboBox);
|
2011-04-27 10:05:43 +00:00
|
|
|
lineEdit->setFocus();
|
2015-07-30 13:16:36 +00:00
|
|
|
QTRY_COMPARE(completer.widget(), lineEdit);
|
2011-04-27 10:05:43 +00:00
|
|
|
comboBox->setFocus();
|
2015-07-30 13:16:36 +00:00
|
|
|
QTRY_COMPARE(completer.widget(), comboBox);
|
2011-04-27 10:05:43 +00:00
|
|
|
lineEdit2->setFocus();
|
2015-07-30 13:16:36 +00:00
|
|
|
QTRY_COMPARE(completer.widget(), comboBox);
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::dynamicSortOrder()
|
|
|
|
{
|
|
|
|
QStandardItemModel model;
|
|
|
|
QCompleter completer(&model);
|
|
|
|
completer.setModelSorting(QCompleter::CaseSensitivelySortedModel);
|
|
|
|
QStandardItem *root = model.invisibleRootItem();
|
|
|
|
for (int i = 0; i < 20; i++) {
|
2015-10-16 07:38:51 +00:00
|
|
|
root->appendRow(new QStandardItem(QString::number(i)));
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
|
|
|
root->appendRow(new QStandardItem("13"));
|
|
|
|
root->sortChildren(0, Qt::AscendingOrder);
|
|
|
|
completer.setCompletionPrefix("1");
|
|
|
|
QCOMPARE(completer.completionCount(), 12);
|
|
|
|
completer.setCompletionPrefix("13");
|
|
|
|
QCOMPARE(completer.completionCount(), 2);
|
|
|
|
|
|
|
|
root->sortChildren(0, Qt::DescendingOrder);
|
|
|
|
completer.setCompletionPrefix("13");
|
|
|
|
QCOMPARE(completer.completionCount(), 2);
|
|
|
|
completer.setCompletionPrefix("1");
|
|
|
|
QCOMPARE(completer.completionCount(), 12);
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::disabledItems()
|
|
|
|
{
|
|
|
|
QLineEdit lineEdit;
|
2019-05-17 09:31:00 +00:00
|
|
|
lineEdit.setWindowTitle(QLatin1String(QTest::currentTestFunction()));
|
|
|
|
auto model = new QStandardItemModel(&lineEdit);
|
2011-04-27 10:05:43 +00:00
|
|
|
QStandardItem *suggestions = new QStandardItem("suggestions");
|
|
|
|
suggestions->setEnabled(false);
|
|
|
|
model->appendRow(suggestions);
|
|
|
|
model->appendRow(new QStandardItem("suggestions Enabled"));
|
2019-05-17 09:31:00 +00:00
|
|
|
auto completer = new QCompleter(model, &lineEdit);
|
2016-12-14 07:01:58 +00:00
|
|
|
QSignalSpy spy(completer, QOverload<const QString &>::of(&QCompleter::activated));
|
2011-04-27 10:05:43 +00:00
|
|
|
lineEdit.setCompleter(completer);
|
2013-04-04 12:09:20 +00:00
|
|
|
lineEdit.move(200, 200);
|
2011-04-27 10:05:43 +00:00
|
|
|
lineEdit.show();
|
2013-04-04 12:09:20 +00:00
|
|
|
QVERIFY(QTest::qWaitForWindowExposed(&lineEdit));
|
2011-04-27 10:05:43 +00:00
|
|
|
|
|
|
|
QTest::keyPress(&lineEdit, Qt::Key_S);
|
|
|
|
QTest::keyPress(&lineEdit, Qt::Key_U);
|
|
|
|
QAbstractItemView *view = lineEdit.completer()->popup();
|
|
|
|
QVERIFY(view->isVisible());
|
2019-05-17 09:31:00 +00:00
|
|
|
QTest::mouseClick(view->viewport(), Qt::LeftButton, {}, view->visualRect(view->model()->index(0, 0)).center());
|
2011-04-27 10:05:43 +00:00
|
|
|
QCOMPARE(spy.count(), 0);
|
|
|
|
QVERIFY(view->isVisible());
|
2019-05-17 09:31:00 +00:00
|
|
|
QTest::mouseClick(view->viewport(), Qt::LeftButton, {}, view->visualRect(view->model()->index(1, 0)).center());
|
2011-04-27 10:05:43 +00:00
|
|
|
QCOMPARE(spy.count(), 1);
|
|
|
|
QVERIFY(!view->isVisible());
|
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::task178797_activatedOnReturn()
|
|
|
|
{
|
2019-10-11 13:21:54 +00:00
|
|
|
if (QGuiApplication::platformName().startsWith(QLatin1String("wayland"), Qt::CaseInsensitive))
|
|
|
|
QSKIP("Wayland: This fails. Figure out why.");
|
|
|
|
|
2013-04-04 12:09:20 +00:00
|
|
|
QLineEdit ledit;
|
|
|
|
setFrameless(&ledit);
|
2019-05-17 09:31:00 +00:00
|
|
|
auto completer = new QCompleter({"foobar1", "foobar2"}, &ledit);
|
2013-04-04 12:09:20 +00:00
|
|
|
ledit.setCompleter(completer);
|
2016-12-14 07:01:58 +00:00
|
|
|
QSignalSpy spy(completer, QOverload<const QString &>::of(&QCompleter::activated));
|
2011-04-27 10:05:43 +00:00
|
|
|
QCOMPARE(spy.count(), 0);
|
2013-04-04 12:09:20 +00:00
|
|
|
ledit.move(200, 200);
|
|
|
|
ledit.show();
|
2019-05-17 09:31:00 +00:00
|
|
|
QApplication::setActiveWindow(&ledit);
|
2013-04-04 12:09:20 +00:00
|
|
|
QVERIFY(QTest::qWaitForWindowActive(&ledit));
|
|
|
|
QTest::keyClick(&ledit, Qt::Key_F);
|
2019-05-17 09:31:00 +00:00
|
|
|
QCoreApplication::processEvents();
|
|
|
|
QTRY_VERIFY(QApplication::activePopupWidget());
|
|
|
|
QTest::keyClick(QApplication::activePopupWidget(), Qt::Key_Down);
|
|
|
|
QCoreApplication::processEvents();
|
|
|
|
QTest::keyClick(QApplication::activePopupWidget(), Qt::Key_Return);
|
|
|
|
QCoreApplication::processEvents();
|
2011-04-27 10:05:43 +00:00
|
|
|
QCOMPARE(spy.count(), 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
class task189564_StringListModel : public QStringListModel
|
|
|
|
{
|
|
|
|
const QString omitString;
|
2019-05-17 09:31:00 +00:00
|
|
|
Qt::ItemFlags flags(const QModelIndex &index) const override
|
2011-04-27 10:05:43 +00:00
|
|
|
{
|
|
|
|
Qt::ItemFlags flags = Qt::ItemIsEnabled;
|
|
|
|
if (data(index, Qt::DisplayRole).toString() != omitString)
|
|
|
|
flags |= Qt::ItemIsSelectable;
|
|
|
|
return flags;
|
|
|
|
}
|
|
|
|
public:
|
2019-05-17 09:31:00 +00:00
|
|
|
explicit task189564_StringListModel(const QString &omitString, QObject *parent = nullptr)
|
2011-04-27 10:05:43 +00:00
|
|
|
: QStringListModel(parent)
|
|
|
|
, omitString(omitString)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
void tst_QCompleter::task189564_omitNonSelectableItems()
|
|
|
|
{
|
|
|
|
const QString prefix("a");
|
|
|
|
const int n = 5;
|
|
|
|
|
|
|
|
QStringList strings;
|
|
|
|
for (int i = 0; i < n; ++i)
|
2015-10-16 07:38:51 +00:00
|
|
|
strings << prefix + QString::number(i);
|
2011-04-27 10:05:43 +00:00
|
|
|
const QString omitString(strings.at(n / 2));
|
|
|
|
task189564_StringListModel model(omitString);
|
|
|
|
model.setStringList(strings);
|
|
|
|
QCompleter completer_(&model);
|
|
|
|
completer_.setCompletionPrefix(prefix);
|
|
|
|
|
|
|
|
QAbstractItemModel *completionModel = completer_.completionModel();
|
|
|
|
QModelIndexList matches1 =
|
|
|
|
completionModel->match(completionModel->index(0, 0), Qt::DisplayRole, prefix, -1);
|
|
|
|
QCOMPARE(matches1.size(), n - 1);
|
|
|
|
QModelIndexList matches2 =
|
|
|
|
completionModel->match(completionModel->index(0, 0), Qt::DisplayRole, omitString);
|
|
|
|
QVERIFY(matches2.isEmpty());
|
|
|
|
}
|
|
|
|
|
|
|
|
class task246056_ComboBox : public QComboBox
|
|
|
|
{
|
|
|
|
Q_OBJECT
|
|
|
|
public:
|
|
|
|
task246056_ComboBox()
|
|
|
|
{
|
|
|
|
setEditable(true);
|
|
|
|
setInsertPolicy(NoInsert);
|
2011-04-20 02:16:58 +00:00
|
|
|
if (completer()) {
|
|
|
|
completer()->setCompletionMode(QCompleter::PopupCompletion);
|
|
|
|
completer()->setCompletionRole(Qt::DisplayRole);
|
2016-12-14 07:01:58 +00:00
|
|
|
connect(lineEdit(), &QLineEdit::editingFinished, this, &task246056_ComboBox::setCompletionPrefix);
|
2011-04-20 02:16:58 +00:00
|
|
|
}
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
|
|
|
private slots:
|
|
|
|
void setCompletionPrefix() { completer()->setCompletionPrefix(lineEdit()->text()); }
|
|
|
|
};
|
|
|
|
|
|
|
|
void tst_QCompleter::task246056_setCompletionPrefix()
|
|
|
|
{
|
2019-10-11 13:21:54 +00:00
|
|
|
if (QGuiApplication::platformName().startsWith(QLatin1String("wayland"), Qt::CaseInsensitive))
|
|
|
|
QSKIP("Wayland: This fails. Figure out why.");
|
|
|
|
|
2013-04-04 12:09:20 +00:00
|
|
|
task246056_ComboBox comboBox;
|
|
|
|
setFrameless(&comboBox);
|
|
|
|
QVERIFY(comboBox.completer());
|
|
|
|
comboBox.addItem("");
|
|
|
|
comboBox.addItem("a1");
|
|
|
|
comboBox.addItem("a2");
|
|
|
|
comboBox.move(200, 200);
|
|
|
|
comboBox.show();
|
|
|
|
QApplication::setActiveWindow(&comboBox);
|
|
|
|
QVERIFY(QTest::qWaitForWindowActive(&comboBox));
|
2016-12-14 07:01:58 +00:00
|
|
|
QSignalSpy spy(comboBox.completer(), QOverload<const QModelIndex &>::of(&QCompleter::activated));
|
2013-04-04 12:09:20 +00:00
|
|
|
QTest::keyPress(&comboBox, 'a');
|
|
|
|
QTest::keyPress(comboBox.completer()->popup(), Qt::Key_Down);
|
|
|
|
QTest::keyPress(comboBox.completer()->popup(), Qt::Key_Down);
|
|
|
|
QTest::keyPress(comboBox.completer()->popup(), Qt::Key_Enter); // don't crash!
|
2013-08-07 12:23:18 +00:00
|
|
|
QCOMPARE(spy.count(), 1);
|
2019-05-17 09:31:00 +00:00
|
|
|
const auto index = spy.at(0).constFirst().toModelIndex();
|
2013-08-07 12:23:18 +00:00
|
|
|
QVERIFY(!index.isValid());
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
class task250064_TextEdit : public QTextEdit
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
QCompleter *completer;
|
|
|
|
|
|
|
|
task250064_TextEdit()
|
|
|
|
{
|
2016-09-28 08:49:57 +00:00
|
|
|
completer = new QCompleter(this);
|
2011-04-27 10:05:43 +00:00
|
|
|
completer->setWidget(this);
|
|
|
|
}
|
|
|
|
|
2019-05-17 09:31:00 +00:00
|
|
|
void keyPressEvent (QKeyEvent *e) override
|
2011-04-27 10:05:43 +00:00
|
|
|
{
|
|
|
|
completer->popup();
|
|
|
|
QTextEdit::keyPressEvent(e);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class task250064_Widget : public QWidget
|
|
|
|
{
|
|
|
|
Q_OBJECT
|
|
|
|
public:
|
2013-04-04 12:09:20 +00:00
|
|
|
task250064_Widget() : m_textEdit(new task250064_TextEdit)
|
2011-04-27 10:05:43 +00:00
|
|
|
{
|
2019-05-17 09:31:00 +00:00
|
|
|
auto tabWidget = new QTabWidget;
|
2011-04-27 10:05:43 +00:00
|
|
|
tabWidget->setFocusPolicy(Qt::ClickFocus);
|
2013-04-04 12:09:20 +00:00
|
|
|
tabWidget->addTab(m_textEdit, "untitled");
|
2011-04-27 10:05:43 +00:00
|
|
|
|
2019-05-17 09:31:00 +00:00
|
|
|
auto layout = new QVBoxLayout(this);
|
2011-04-27 10:05:43 +00:00
|
|
|
layout->addWidget(tabWidget);
|
|
|
|
|
2013-04-04 12:09:20 +00:00
|
|
|
m_textEdit->setPlainText("bla bla bla");
|
|
|
|
m_textEdit->setFocus();
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void setCompletionModel()
|
|
|
|
{
|
2019-05-17 09:31:00 +00:00
|
|
|
m_textEdit->completer->setModel(nullptr);
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
2013-04-04 12:09:20 +00:00
|
|
|
|
|
|
|
QTextEdit *textEdit() const { return m_textEdit; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
task250064_TextEdit *m_textEdit;
|
2011-04-27 10:05:43 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
void tst_QCompleter::task250064_lostFocus()
|
|
|
|
{
|
2019-10-11 13:21:54 +00:00
|
|
|
if (QGuiApplication::platformName().startsWith(QLatin1String("wayland"), Qt::CaseInsensitive))
|
|
|
|
QSKIP("Wayland: This fails. Figure out why.");
|
|
|
|
|
2013-04-04 12:09:20 +00:00
|
|
|
task250064_Widget widget;
|
2019-05-17 09:31:00 +00:00
|
|
|
widget.setWindowTitle(QLatin1String(QTest::currentTestFunction()));
|
2013-04-04 12:09:20 +00:00
|
|
|
widget.show();
|
|
|
|
QApplication::setActiveWindow(&widget);
|
|
|
|
QVERIFY(QTest::qWaitForWindowActive(&widget));
|
|
|
|
QTest::keyPress(widget.textEdit(), 'a');
|
|
|
|
Qt::FocusPolicy origPolicy = widget.textEdit()->focusPolicy();
|
2011-04-27 10:05:43 +00:00
|
|
|
QVERIFY(origPolicy != Qt::NoFocus);
|
2013-04-04 12:09:20 +00:00
|
|
|
widget.setCompletionModel();
|
|
|
|
QCOMPARE(widget.textEdit()->focusPolicy(), origPolicy);
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::task253125_lineEditCompletion_data()
|
|
|
|
{
|
|
|
|
QTest::addColumn<QStringList>("list");
|
2019-05-17 09:31:00 +00:00
|
|
|
QTest::addColumn<QCompleter::CompletionMode>("completionMode");
|
2011-04-27 10:05:43 +00:00
|
|
|
|
2019-05-17 09:31:00 +00:00
|
|
|
QStringList list = {"alpha", "beta", "gamma", "delta", "epsilon", "zeta",
|
|
|
|
"eta", "theta", "iota", "kappa", "lambda", "mu",
|
|
|
|
"nu", "xi", "omicron", "pi", "rho", "sigma",
|
|
|
|
"tau", "upsilon", "phi", "chi", "psi", "omega"};
|
2011-04-27 10:05:43 +00:00
|
|
|
|
2019-05-17 09:31:00 +00:00
|
|
|
QTest::newRow("Inline") << list << QCompleter::InlineCompletion;
|
|
|
|
QTest::newRow("Filtered") << list << QCompleter::PopupCompletion;
|
|
|
|
QTest::newRow("Unfiltered") << list << QCompleter::UnfilteredPopupCompletion;
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::task253125_lineEditCompletion()
|
|
|
|
{
|
2019-10-11 13:21:54 +00:00
|
|
|
if (QGuiApplication::platformName().startsWith(QLatin1String("wayland"), Qt::CaseInsensitive))
|
|
|
|
QSKIP("Wayland: This fails. Figure out why.");
|
|
|
|
|
2011-04-27 10:05:43 +00:00
|
|
|
QFETCH(QStringList, list);
|
2019-05-17 09:31:00 +00:00
|
|
|
QFETCH(QCompleter::CompletionMode, completionMode);
|
2011-04-27 10:05:43 +00:00
|
|
|
|
2019-05-17 09:31:00 +00:00
|
|
|
std::unique_ptr<QStringListModel> model(new QStringListModel(list));
|
2011-04-27 10:05:43 +00:00
|
|
|
|
2019-05-17 09:31:00 +00:00
|
|
|
std::unique_ptr<QCompleter> completer(new QCompleter(list));
|
|
|
|
completer->setModel(model.get());
|
|
|
|
completer->setCompletionMode(completionMode);
|
2011-04-27 10:05:43 +00:00
|
|
|
|
|
|
|
QLineEdit edit;
|
2019-05-17 09:31:00 +00:00
|
|
|
edit.setWindowTitle(QLatin1String(QTest::currentTestFunction()) + QLatin1String("::")
|
|
|
|
+ QLatin1String(QTest::currentDataTag()));
|
|
|
|
edit.setCompleter(completer.get());
|
2013-04-04 12:09:20 +00:00
|
|
|
edit.move(200, 200);
|
2011-04-27 10:05:43 +00:00
|
|
|
edit.show();
|
|
|
|
edit.setFocus();
|
|
|
|
QApplication::setActiveWindow(&edit);
|
2012-07-18 11:12:59 +00:00
|
|
|
QVERIFY(QTest::qWaitForWindowActive(&edit));
|
2011-04-27 10:05:43 +00:00
|
|
|
|
|
|
|
QTest::keyClick(&edit, 'i');
|
|
|
|
QCOMPARE(edit.completer()->currentCompletion(), QString("iota"));
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Down);
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Enter);
|
|
|
|
|
|
|
|
QCOMPARE(edit.text(), QString("iota"));
|
|
|
|
|
2012-12-31 10:11:25 +00:00
|
|
|
edit.clear();
|
|
|
|
completer->setCompletionMode(QCompleter::PopupCompletion);
|
|
|
|
completer->setFilterMode(Qt::MatchContains);
|
|
|
|
|
|
|
|
QTest::keyClick(&edit, 't');
|
|
|
|
QCOMPARE(edit.completer()->currentCompletion(), QString("beta"));
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Down);
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Enter);
|
|
|
|
|
|
|
|
QCOMPARE(edit.text(), QString("beta"));
|
|
|
|
|
|
|
|
edit.clear();
|
|
|
|
|
|
|
|
QTest::keyClick(&edit, 'p');
|
|
|
|
QTest::keyClick(&edit, 'p');
|
|
|
|
QCOMPARE(edit.completer()->currentCompletion(), QString("kappa"));
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Down);
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Enter);
|
|
|
|
|
|
|
|
QCOMPARE(edit.text(), QString("kappa"));
|
|
|
|
|
|
|
|
edit.clear();
|
|
|
|
completer->setFilterMode(Qt::MatchStartsWith);
|
|
|
|
|
|
|
|
QTest::keyClick(&edit, 't');
|
|
|
|
QCOMPARE(edit.completer()->currentCompletion(), QString("theta"));
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Down);
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Enter);
|
|
|
|
|
|
|
|
QCOMPARE(edit.text(), QString("theta"));
|
|
|
|
|
|
|
|
edit.clear();
|
|
|
|
|
|
|
|
QTest::keyClick(&edit, 'p');
|
|
|
|
QTest::keyClick(&edit, 'p');
|
|
|
|
QCOMPARE(edit.completer()->currentCompletion(), QString());
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Down);
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Enter);
|
|
|
|
|
|
|
|
QCOMPARE(edit.text(), QString("pp"));
|
|
|
|
|
|
|
|
edit.clear();
|
|
|
|
|
|
|
|
QTest::keyClick(&edit, 'u');
|
|
|
|
QTest::keyClick(&edit, 'p');
|
|
|
|
QTest::keyClick(&edit, 's');
|
|
|
|
QCOMPARE(edit.completer()->currentCompletion(), QString("upsilon"));
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Down);
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Enter);
|
|
|
|
|
|
|
|
QCOMPARE(edit.text(), QString("upsilon"));
|
|
|
|
|
|
|
|
edit.clear();
|
|
|
|
completer->setFilterMode(Qt::MatchEndsWith);
|
|
|
|
|
|
|
|
QTest::keyClick(&edit, 'm');
|
|
|
|
QTest::keyClick(&edit, 'm');
|
|
|
|
QTest::keyClick(&edit, 'a');
|
|
|
|
QCOMPARE(edit.completer()->currentCompletion(), QString("gamma"));
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Down);
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Enter);
|
|
|
|
|
|
|
|
QCOMPARE(edit.text(), QString("gamma"));
|
|
|
|
|
|
|
|
edit.clear();
|
|
|
|
|
|
|
|
QTest::keyClick(&edit, 'g');
|
|
|
|
QTest::keyClick(&edit, 'm');
|
|
|
|
QTest::keyClick(&edit, 'a');
|
|
|
|
QCOMPARE(edit.completer()->currentCompletion(), QString("sigma"));
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Down);
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Enter);
|
|
|
|
|
|
|
|
QCOMPARE(edit.text(), QString("sigma"));
|
|
|
|
|
|
|
|
edit.clear();
|
|
|
|
|
|
|
|
QTest::keyClick(&edit, 'm');
|
|
|
|
QTest::keyClick(&edit, 'm');
|
|
|
|
QCOMPARE(edit.completer()->currentCompletion(), QString());
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Down);
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Enter);
|
|
|
|
|
|
|
|
QCOMPARE(edit.text(), QString("mm"));
|
|
|
|
|
|
|
|
edit.clear();
|
|
|
|
completer->setFilterMode(Qt::MatchStartsWith);
|
|
|
|
|
|
|
|
QTest::keyClick(&edit, 'z');
|
|
|
|
QTest::keyClick(&edit, 'e');
|
|
|
|
QCOMPARE(edit.completer()->currentCompletion(), QString("zeta"));
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Down);
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Enter);
|
|
|
|
|
|
|
|
QCOMPARE(edit.text(), QString("zeta"));
|
|
|
|
|
|
|
|
edit.clear();
|
|
|
|
completer->setFilterMode(Qt::MatchEndsWith);
|
|
|
|
|
|
|
|
QTest::keyClick(&edit, 'e');
|
|
|
|
QTest::keyClick(&edit, 'g');
|
|
|
|
QTest::keyClick(&edit, 'a');
|
|
|
|
QCOMPARE(edit.completer()->currentCompletion(), QString("omega"));
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Down);
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Enter);
|
|
|
|
|
|
|
|
QCOMPARE(edit.text(), QString("omega"));
|
|
|
|
|
|
|
|
edit.clear();
|
|
|
|
completer->setFilterMode(Qt::MatchContains);
|
|
|
|
|
|
|
|
QTest::keyClick(&edit, 'c');
|
|
|
|
QTest::keyClick(&edit, 'r');
|
|
|
|
QCOMPARE(edit.completer()->currentCompletion(), QString("omicron"));
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Down);
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Enter);
|
|
|
|
|
|
|
|
QCOMPARE(edit.text(), QString("omicron"));
|
|
|
|
|
|
|
|
edit.clear();
|
|
|
|
|
|
|
|
QTest::keyClick(&edit, 'z');
|
|
|
|
QTest::keyClick(&edit, 'z');
|
|
|
|
QCOMPARE(edit.completer()->currentCompletion(), QString());
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Down);
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Enter);
|
|
|
|
|
|
|
|
QCOMPARE(edit.text(), QString("zz"));
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::task247560_keyboardNavigation()
|
|
|
|
{
|
2019-10-11 13:21:54 +00:00
|
|
|
if (QGuiApplication::platformName().startsWith(QLatin1String("wayland"), Qt::CaseInsensitive))
|
|
|
|
QSKIP("Wayland: This fails. Figure out why.");
|
|
|
|
|
2011-04-27 10:05:43 +00:00
|
|
|
QStandardItemModel model;
|
|
|
|
|
|
|
|
for (int i = 0; i < 5; i++) {
|
2015-10-16 07:38:51 +00:00
|
|
|
const QString prefix = QLatin1String("row ") + QString::number(i) + QLatin1String(" column ");
|
|
|
|
for (int j = 0; j < 5; j++)
|
|
|
|
model.setItem(i, j, new QStandardItem(prefix + QString::number(j)));
|
2011-04-27 10:05:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
QCompleter completer(&model);
|
|
|
|
completer.setCompletionColumn(1);
|
|
|
|
|
|
|
|
QLineEdit edit;
|
2019-05-17 09:31:00 +00:00
|
|
|
edit.setWindowTitle(QLatin1String(QTest::currentTestFunction()));
|
2011-04-27 10:05:43 +00:00
|
|
|
edit.setCompleter(&completer);
|
2013-04-04 12:09:20 +00:00
|
|
|
edit.move(200, 200);
|
2011-04-27 10:05:43 +00:00
|
|
|
edit.show();
|
|
|
|
edit.setFocus();
|
|
|
|
QApplication::setActiveWindow(&edit);
|
2012-07-18 11:12:59 +00:00
|
|
|
QVERIFY(QTest::qWaitForWindowActive(&edit));
|
2011-04-27 10:05:43 +00:00
|
|
|
|
|
|
|
QTest::keyClick(&edit, 'r');
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Down);
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Down);
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Enter);
|
|
|
|
|
|
|
|
QCOMPARE(edit.text(), QString("row 1 column 1"));
|
|
|
|
|
|
|
|
edit.clear();
|
|
|
|
|
|
|
|
QTest::keyClick(&edit, 'r');
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Up);
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Up);
|
|
|
|
QTest::keyClick(edit.completer()->popup(), Qt::Key_Enter);
|
|
|
|
|
|
|
|
QCOMPARE(edit.text(), QString("row 3 column 1"));
|
|
|
|
}
|
|
|
|
|
2012-07-25 13:18:19 +00:00
|
|
|
// Helpers for QTBUG_14292_filesystem: Recursion helper for below recurseTreeModel
|
|
|
|
// Function to recurse over a tree model applying a function
|
|
|
|
// taking index and depth, returning true to terminate recursion.
|
|
|
|
template <class Function>
|
2019-02-06 21:09:33 +00:00
|
|
|
bool recurseTreeModel(const QAbstractItemModel &m, const QModelIndex &idx, Function f, int depth = 0)
|
2012-07-25 13:18:19 +00:00
|
|
|
{
|
2019-02-06 21:09:33 +00:00
|
|
|
if (idx.isValid() && f(idx, depth))
|
|
|
|
return true;
|
|
|
|
const int rowCount = m.rowCount(idx);
|
2012-07-25 13:18:19 +00:00
|
|
|
for (int row = 0; row < rowCount; ++row)
|
2019-02-06 21:09:33 +00:00
|
|
|
if (recurseTreeModel(m, m.index(row, 0, idx), f, depth + 1))
|
2012-07-25 13:18:19 +00:00
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Function applicable to the above recurseTreeModel() to search for a data item.
|
|
|
|
class SearchFunction
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
SearchFunction(const QString &needle, int role = Qt::DisplayRole) :
|
|
|
|
m_needle(needle), m_role(role) {}
|
|
|
|
|
|
|
|
bool operator()(const QModelIndex &idx, int /* depth */) const
|
|
|
|
{ return idx.data(m_role).toString() == m_needle; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
const QString m_needle;
|
|
|
|
const int m_role;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Function applicable to the above recurseTreeModel() for debug output
|
|
|
|
// of a model.
|
|
|
|
class DebugFunction
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
DebugFunction(QDebug d) : m_d(d) {}
|
|
|
|
|
|
|
|
bool operator()(const QModelIndex &idx, int depth)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < 4 * depth; ++i)
|
|
|
|
m_d << ' ';
|
|
|
|
m_d << idx.data(QFileSystemModel::FileNameRole).toString()
|
|
|
|
<< '\n';
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
private:
|
|
|
|
QDebug m_d;
|
|
|
|
};
|
|
|
|
|
|
|
|
QDebug operator<<(QDebug d, const QAbstractItemModel &m)
|
|
|
|
{
|
|
|
|
QDebug dns = d.nospace();
|
|
|
|
dns << '\n';
|
2019-02-06 21:09:33 +00:00
|
|
|
recurseTreeModel(m, QModelIndex(), DebugFunction(dns));
|
2012-07-25 13:18:19 +00:00
|
|
|
return d;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char testDir1[] = "hello";
|
|
|
|
static const char testDir2[] = "holla";
|
|
|
|
|
|
|
|
// Helper for QTBUG_14292_filesystem, checking whether both
|
|
|
|
// test directories are seen by the file system model for usage
|
|
|
|
// with QTRY_VERIFY.
|
|
|
|
|
|
|
|
static inline bool testFileSystemReady(const QAbstractItemModel &model)
|
|
|
|
{
|
2019-02-06 21:09:33 +00:00
|
|
|
return recurseTreeModel(model, QModelIndex(), SearchFunction(QLatin1String(testDir1), QFileSystemModel::FileNameRole))
|
|
|
|
&& recurseTreeModel(model, QModelIndex(), SearchFunction(QLatin1String(testDir2), QFileSystemModel::FileNameRole));
|
2012-07-25 13:18:19 +00:00
|
|
|
}
|
|
|
|
|
2011-04-27 10:05:43 +00:00
|
|
|
void tst_QCompleter::QTBUG_14292_filesystem()
|
|
|
|
{
|
2019-10-11 13:21:54 +00:00
|
|
|
if (QGuiApplication::platformName().startsWith(QLatin1String("wayland"), Qt::CaseInsensitive))
|
|
|
|
QSKIP("Wayland: This fails. Figure out why.");
|
|
|
|
|
2012-07-25 13:18:19 +00:00
|
|
|
// This test tests whether the creation of subdirectories
|
|
|
|
// does not cause completers based on file system models
|
|
|
|
// to pop up the completion list due to file changed signals.
|
2011-04-27 10:05:43 +00:00
|
|
|
FileSystem fs;
|
2012-07-25 13:18:19 +00:00
|
|
|
QFileSystemModel model;
|
|
|
|
model.setRootPath(fs.path());
|
2011-04-27 10:05:43 +00:00
|
|
|
|
2012-07-25 13:18:19 +00:00
|
|
|
QVERIFY(fs.createDirectory(QLatin1String(testDir1)));
|
|
|
|
QVERIFY(fs.createDirectory(QLatin1String(testDir2)));
|
2011-04-27 10:05:43 +00:00
|
|
|
|
|
|
|
QLineEdit edit;
|
2019-05-17 09:31:00 +00:00
|
|
|
edit.setWindowTitle(QLatin1String(QTest::currentTestFunction()));
|
2011-04-27 10:05:43 +00:00
|
|
|
QCompleter comp;
|
|
|
|
comp.setModel(&model);
|
|
|
|
edit.setCompleter(&comp);
|
|
|
|
|
2013-04-04 12:09:20 +00:00
|
|
|
edit.move(200, 200);
|
2011-04-27 10:05:43 +00:00
|
|
|
edit.show();
|
|
|
|
QApplication::setActiveWindow(&edit);
|
2012-07-18 11:12:59 +00:00
|
|
|
QVERIFY(QTest::qWaitForWindowActive(&edit));
|
|
|
|
QCOMPARE(QApplication::activeWindow(), &edit);
|
2011-04-27 10:05:43 +00:00
|
|
|
edit.setFocus();
|
|
|
|
QTRY_VERIFY(edit.hasFocus());
|
|
|
|
|
2012-07-25 13:18:19 +00:00
|
|
|
// Wait for all file system model slots/timers to trigger
|
|
|
|
// until the model sees the subdirectories.
|
|
|
|
QTRY_VERIFY(testFileSystemReady(model));
|
|
|
|
// But this should not cause the combo to pop up.
|
2011-04-27 10:05:43 +00:00
|
|
|
QVERIFY(!comp.popup()->isVisible());
|
2012-07-25 09:47:26 +00:00
|
|
|
edit.setText(fs.path());
|
2011-04-27 10:05:43 +00:00
|
|
|
QTest::keyClick(&edit, '/');
|
|
|
|
QTRY_VERIFY(comp.popup()->isVisible());
|
|
|
|
QCOMPARE(comp.popup()->model()->rowCount(), 2);
|
|
|
|
QApplication::processEvents();
|
|
|
|
QTest::keyClick(&edit, 'h');
|
|
|
|
QCOMPARE(comp.popup()->model()->rowCount(), 2);
|
|
|
|
QTest::keyClick(&edit, 'e');
|
|
|
|
QCOMPARE(comp.popup()->model()->rowCount(), 1);
|
|
|
|
QTest::keyClick(&edit, 'r');
|
|
|
|
QTRY_VERIFY(!comp.popup()->isVisible());
|
2012-07-25 09:47:26 +00:00
|
|
|
QVERIFY(fs.createDirectory(QStringLiteral("hero")));
|
2011-04-27 10:05:43 +00:00
|
|
|
QTRY_VERIFY(comp.popup()->isVisible());
|
|
|
|
QCOMPARE(comp.popup()->model()->rowCount(), 1);
|
|
|
|
QTest::keyClick(comp.popup(), Qt::Key_Escape);
|
|
|
|
QTRY_VERIFY(!comp.popup()->isVisible());
|
2012-07-25 09:47:26 +00:00
|
|
|
QVERIFY(fs.createDirectory(QStringLiteral("nothingThere")));
|
2011-04-27 10:05:43 +00:00
|
|
|
//there is no reason creating a file should open a popup, it did in Qt 4.7.0
|
|
|
|
QTest::qWait(60);
|
|
|
|
QVERIFY(!comp.popup()->isVisible());
|
|
|
|
|
|
|
|
QTest::keyClick(&edit, Qt::Key_Backspace);
|
|
|
|
QTRY_VERIFY(comp.popup()->isVisible());
|
|
|
|
QCOMPARE(comp.popup()->model()->rowCount(), 2);
|
|
|
|
QTest::keyClick(&edit, 'm');
|
|
|
|
QTRY_VERIFY(!comp.popup()->isVisible());
|
|
|
|
|
|
|
|
QWidget w;
|
2013-04-04 12:09:20 +00:00
|
|
|
w.move(400, 200);
|
2011-04-27 10:05:43 +00:00
|
|
|
w.show();
|
|
|
|
QApplication::setActiveWindow(&w);
|
2012-07-24 12:29:01 +00:00
|
|
|
QVERIFY(QTest::qWaitForWindowActive(&w));
|
|
|
|
QVERIFY(!edit.hasFocus() && !comp.popup()->hasFocus());
|
2011-04-27 10:05:43 +00:00
|
|
|
|
2012-07-25 09:47:26 +00:00
|
|
|
QVERIFY(fs.createDirectory(QStringLiteral("hemo")));
|
2011-04-27 10:05:43 +00:00
|
|
|
//there is no reason creating a file should open a popup, it did in Qt 4.7.0
|
|
|
|
QTest::qWait(60);
|
|
|
|
QVERIFY(!comp.popup()->isVisible());
|
|
|
|
}
|
|
|
|
|
QCompleter: Send activated() signal only once on return key
Due to the complex event forwarding logic between QCompleter,
QComboBox, QLineEdit and QWidgetLineControl, in some cases the
same single user return key press could result in duplicated
activated() signals being emitted by QComboBox. The first one
would be emitted because QLineEdit emitted editingFinished()
as a result of QCompleter::eventFilter() having forwarded the
return key press event to QComboBox. The second one, would
happen right after, as QCompleter::eventFilter() would process
the same event on behalf of its popup.
(We recall that QCompleter is installed as its own popup event
filter. That's also the case for the completer's widget, although
the purpose there is limited to focus-out events).
The current fix consists on skipping the emit as a result of
QLineEdit::editingFinished() if the completer's popup is still
active. For this to be accurate, it helps to test whether the
completer's popup is visible, so we will not be hiding it in
QWidgetLineControl::processKeyEvent() anymore. Indeed, we know
that if the popup is visible, that means that processKeyEvent()
was called after being forwarded by the completer's popup event
filter. Furthermore, the popup will be hidden by its event filter
shortly after it returns from said event forwarding call.
Based on a patch by Alexey Chernov <4ernov@gmail.com>.
Task-number: QTBUG-51858
Task-number: QTBUG-51889
Change-Id: I013f6c3000ae37b5b0ec20eaf5cf7746c9c903e3
Reviewed-by: Richard Moe Gustavsen <richard.gustavsen@qt.io>
2017-08-28 03:04:35 +00:00
|
|
|
void tst_QCompleter::QTBUG_52028_tabAutoCompletes()
|
|
|
|
{
|
2019-10-11 13:21:54 +00:00
|
|
|
if (QGuiApplication::platformName().startsWith(QLatin1String("wayland"), Qt::CaseInsensitive))
|
|
|
|
QSKIP("Wayland: This fails. Figure out why.");
|
|
|
|
|
QCompleter: Send activated() signal only once on return key
Due to the complex event forwarding logic between QCompleter,
QComboBox, QLineEdit and QWidgetLineControl, in some cases the
same single user return key press could result in duplicated
activated() signals being emitted by QComboBox. The first one
would be emitted because QLineEdit emitted editingFinished()
as a result of QCompleter::eventFilter() having forwarded the
return key press event to QComboBox. The second one, would
happen right after, as QCompleter::eventFilter() would process
the same event on behalf of its popup.
(We recall that QCompleter is installed as its own popup event
filter. That's also the case for the completer's widget, although
the purpose there is limited to focus-out events).
The current fix consists on skipping the emit as a result of
QLineEdit::editingFinished() if the completer's popup is still
active. For this to be accurate, it helps to test whether the
completer's popup is visible, so we will not be hiding it in
QWidgetLineControl::processKeyEvent() anymore. Indeed, we know
that if the popup is visible, that means that processKeyEvent()
was called after being forwarded by the completer's popup event
filter. Furthermore, the popup will be hidden by its event filter
shortly after it returns from said event forwarding call.
Based on a patch by Alexey Chernov <4ernov@gmail.com>.
Task-number: QTBUG-51858
Task-number: QTBUG-51889
Change-Id: I013f6c3000ae37b5b0ec20eaf5cf7746c9c903e3
Reviewed-by: Richard Moe Gustavsen <richard.gustavsen@qt.io>
2017-08-28 03:04:35 +00:00
|
|
|
QWidget w;
|
2019-05-17 09:31:00 +00:00
|
|
|
w.setWindowTitle(QLatin1String(QTest::currentTestFunction()));
|
QCompleter: Send activated() signal only once on return key
Due to the complex event forwarding logic between QCompleter,
QComboBox, QLineEdit and QWidgetLineControl, in some cases the
same single user return key press could result in duplicated
activated() signals being emitted by QComboBox. The first one
would be emitted because QLineEdit emitted editingFinished()
as a result of QCompleter::eventFilter() having forwarded the
return key press event to QComboBox. The second one, would
happen right after, as QCompleter::eventFilter() would process
the same event on behalf of its popup.
(We recall that QCompleter is installed as its own popup event
filter. That's also the case for the completer's widget, although
the purpose there is limited to focus-out events).
The current fix consists on skipping the emit as a result of
QLineEdit::editingFinished() if the completer's popup is still
active. For this to be accurate, it helps to test whether the
completer's popup is visible, so we will not be hiding it in
QWidgetLineControl::processKeyEvent() anymore. Indeed, we know
that if the popup is visible, that means that processKeyEvent()
was called after being forwarded by the completer's popup event
filter. Furthermore, the popup will be hidden by its event filter
shortly after it returns from said event forwarding call.
Based on a patch by Alexey Chernov <4ernov@gmail.com>.
Task-number: QTBUG-51858
Task-number: QTBUG-51889
Change-Id: I013f6c3000ae37b5b0ec20eaf5cf7746c9c903e3
Reviewed-by: Richard Moe Gustavsen <richard.gustavsen@qt.io>
2017-08-28 03:04:35 +00:00
|
|
|
w.setLayout(new QVBoxLayout);
|
|
|
|
|
|
|
|
QComboBox cbox;
|
|
|
|
cbox.setEditable(true);
|
|
|
|
cbox.setInsertPolicy(QComboBox::NoInsert);
|
2019-05-17 09:31:00 +00:00
|
|
|
cbox.addItems({"foobar1", "foobar2", "hux"});
|
QCompleter: Send activated() signal only once on return key
Due to the complex event forwarding logic between QCompleter,
QComboBox, QLineEdit and QWidgetLineControl, in some cases the
same single user return key press could result in duplicated
activated() signals being emitted by QComboBox. The first one
would be emitted because QLineEdit emitted editingFinished()
as a result of QCompleter::eventFilter() having forwarded the
return key press event to QComboBox. The second one, would
happen right after, as QCompleter::eventFilter() would process
the same event on behalf of its popup.
(We recall that QCompleter is installed as its own popup event
filter. That's also the case for the completer's widget, although
the purpose there is limited to focus-out events).
The current fix consists on skipping the emit as a result of
QLineEdit::editingFinished() if the completer's popup is still
active. For this to be accurate, it helps to test whether the
completer's popup is visible, so we will not be hiding it in
QWidgetLineControl::processKeyEvent() anymore. Indeed, we know
that if the popup is visible, that means that processKeyEvent()
was called after being forwarded by the completer's popup event
filter. Furthermore, the popup will be hidden by its event filter
shortly after it returns from said event forwarding call.
Based on a patch by Alexey Chernov <4ernov@gmail.com>.
Task-number: QTBUG-51858
Task-number: QTBUG-51889
Change-Id: I013f6c3000ae37b5b0ec20eaf5cf7746c9c903e3
Reviewed-by: Richard Moe Gustavsen <richard.gustavsen@qt.io>
2017-08-28 03:04:35 +00:00
|
|
|
|
|
|
|
cbox.completer()->setCaseSensitivity(Qt::CaseInsensitive);
|
|
|
|
cbox.completer()->setCompletionMode(QCompleter::PopupCompletion);
|
|
|
|
|
|
|
|
w.layout()->addWidget(&cbox);
|
|
|
|
|
|
|
|
// Adding a line edit is a good reason for tab to do something unrelated
|
2019-05-17 09:31:00 +00:00
|
|
|
auto le = new QLineEdit;
|
|
|
|
w.layout()->addWidget(le);
|
QCompleter: Send activated() signal only once on return key
Due to the complex event forwarding logic between QCompleter,
QComboBox, QLineEdit and QWidgetLineControl, in some cases the
same single user return key press could result in duplicated
activated() signals being emitted by QComboBox. The first one
would be emitted because QLineEdit emitted editingFinished()
as a result of QCompleter::eventFilter() having forwarded the
return key press event to QComboBox. The second one, would
happen right after, as QCompleter::eventFilter() would process
the same event on behalf of its popup.
(We recall that QCompleter is installed as its own popup event
filter. That's also the case for the completer's widget, although
the purpose there is limited to focus-out events).
The current fix consists on skipping the emit as a result of
QLineEdit::editingFinished() if the completer's popup is still
active. For this to be accurate, it helps to test whether the
completer's popup is visible, so we will not be hiding it in
QWidgetLineControl::processKeyEvent() anymore. Indeed, we know
that if the popup is visible, that means that processKeyEvent()
was called after being forwarded by the completer's popup event
filter. Furthermore, the popup will be hidden by its event filter
shortly after it returns from said event forwarding call.
Based on a patch by Alexey Chernov <4ernov@gmail.com>.
Task-number: QTBUG-51858
Task-number: QTBUG-51889
Change-Id: I013f6c3000ae37b5b0ec20eaf5cf7746c9c903e3
Reviewed-by: Richard Moe Gustavsen <richard.gustavsen@qt.io>
2017-08-28 03:04:35 +00:00
|
|
|
|
2019-09-09 14:11:48 +00:00
|
|
|
const auto pos = w.screen()->availableGeometry().topLeft() + QPoint(200,200);
|
QCompleter: Send activated() signal only once on return key
Due to the complex event forwarding logic between QCompleter,
QComboBox, QLineEdit and QWidgetLineControl, in some cases the
same single user return key press could result in duplicated
activated() signals being emitted by QComboBox. The first one
would be emitted because QLineEdit emitted editingFinished()
as a result of QCompleter::eventFilter() having forwarded the
return key press event to QComboBox. The second one, would
happen right after, as QCompleter::eventFilter() would process
the same event on behalf of its popup.
(We recall that QCompleter is installed as its own popup event
filter. That's also the case for the completer's widget, although
the purpose there is limited to focus-out events).
The current fix consists on skipping the emit as a result of
QLineEdit::editingFinished() if the completer's popup is still
active. For this to be accurate, it helps to test whether the
completer's popup is visible, so we will not be hiding it in
QWidgetLineControl::processKeyEvent() anymore. Indeed, we know
that if the popup is visible, that means that processKeyEvent()
was called after being forwarded by the completer's popup event
filter. Furthermore, the popup will be hidden by its event filter
shortly after it returns from said event forwarding call.
Based on a patch by Alexey Chernov <4ernov@gmail.com>.
Task-number: QTBUG-51858
Task-number: QTBUG-51889
Change-Id: I013f6c3000ae37b5b0ec20eaf5cf7746c9c903e3
Reviewed-by: Richard Moe Gustavsen <richard.gustavsen@qt.io>
2017-08-28 03:04:35 +00:00
|
|
|
w.move(pos);
|
|
|
|
w.show();
|
|
|
|
QApplication::setActiveWindow(&w);
|
|
|
|
QVERIFY(QTest::qWaitForWindowActive(&w));
|
|
|
|
|
2020-02-04 14:31:25 +00:00
|
|
|
QSignalSpy activatedSpy(&cbox, &QComboBox::activated);
|
QCompleter: Send activated() signal only once on return key
Due to the complex event forwarding logic between QCompleter,
QComboBox, QLineEdit and QWidgetLineControl, in some cases the
same single user return key press could result in duplicated
activated() signals being emitted by QComboBox. The first one
would be emitted because QLineEdit emitted editingFinished()
as a result of QCompleter::eventFilter() having forwarded the
return key press event to QComboBox. The second one, would
happen right after, as QCompleter::eventFilter() would process
the same event on behalf of its popup.
(We recall that QCompleter is installed as its own popup event
filter. That's also the case for the completer's widget, although
the purpose there is limited to focus-out events).
The current fix consists on skipping the emit as a result of
QLineEdit::editingFinished() if the completer's popup is still
active. For this to be accurate, it helps to test whether the
completer's popup is visible, so we will not be hiding it in
QWidgetLineControl::processKeyEvent() anymore. Indeed, we know
that if the popup is visible, that means that processKeyEvent()
was called after being forwarded by the completer's popup event
filter. Furthermore, the popup will be hidden by its event filter
shortly after it returns from said event forwarding call.
Based on a patch by Alexey Chernov <4ernov@gmail.com>.
Task-number: QTBUG-51858
Task-number: QTBUG-51889
Change-Id: I013f6c3000ae37b5b0ec20eaf5cf7746c9c903e3
Reviewed-by: Richard Moe Gustavsen <richard.gustavsen@qt.io>
2017-08-28 03:04:35 +00:00
|
|
|
|
|
|
|
// Tab key will complete but not activate
|
|
|
|
cbox.lineEdit()->clear();
|
|
|
|
QTest::keyClick(&cbox, Qt::Key_H);
|
|
|
|
QVERIFY(cbox.completer()->popup());
|
|
|
|
QTRY_VERIFY(cbox.completer()->popup()->isVisible());
|
|
|
|
QTest::keyClick(cbox.completer()->popup(), Qt::Key_Tab);
|
|
|
|
QCOMPARE(cbox.completer()->currentCompletion(), QLatin1String("hux"));
|
|
|
|
QCOMPARE(activatedSpy.count(), 0);
|
|
|
|
QEXPECT_FAIL("", "QTBUG-52028 will not be fixed today.", Abort);
|
|
|
|
QCOMPARE(cbox.currentText(), QLatin1String("hux"));
|
|
|
|
QCOMPARE(activatedSpy.count(), 0);
|
2019-05-17 09:31:00 +00:00
|
|
|
QVERIFY(!le->hasFocus());
|
QCompleter: Send activated() signal only once on return key
Due to the complex event forwarding logic between QCompleter,
QComboBox, QLineEdit and QWidgetLineControl, in some cases the
same single user return key press could result in duplicated
activated() signals being emitted by QComboBox. The first one
would be emitted because QLineEdit emitted editingFinished()
as a result of QCompleter::eventFilter() having forwarded the
return key press event to QComboBox. The second one, would
happen right after, as QCompleter::eventFilter() would process
the same event on behalf of its popup.
(We recall that QCompleter is installed as its own popup event
filter. That's also the case for the completer's widget, although
the purpose there is limited to focus-out events).
The current fix consists on skipping the emit as a result of
QLineEdit::editingFinished() if the completer's popup is still
active. For this to be accurate, it helps to test whether the
completer's popup is visible, so we will not be hiding it in
QWidgetLineControl::processKeyEvent() anymore. Indeed, we know
that if the popup is visible, that means that processKeyEvent()
was called after being forwarded by the completer's popup event
filter. Furthermore, the popup will be hidden by its event filter
shortly after it returns from said event forwarding call.
Based on a patch by Alexey Chernov <4ernov@gmail.com>.
Task-number: QTBUG-51858
Task-number: QTBUG-51889
Change-Id: I013f6c3000ae37b5b0ec20eaf5cf7746c9c903e3
Reviewed-by: Richard Moe Gustavsen <richard.gustavsen@qt.io>
2017-08-28 03:04:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void tst_QCompleter::QTBUG_51889_activatedSentTwice()
|
|
|
|
{
|
2019-10-11 13:21:54 +00:00
|
|
|
if (QGuiApplication::platformName().startsWith(QLatin1String("wayland"), Qt::CaseInsensitive))
|
|
|
|
QSKIP("Wayland: This fails. Figure out why.");
|
|
|
|
|
QCompleter: Send activated() signal only once on return key
Due to the complex event forwarding logic between QCompleter,
QComboBox, QLineEdit and QWidgetLineControl, in some cases the
same single user return key press could result in duplicated
activated() signals being emitted by QComboBox. The first one
would be emitted because QLineEdit emitted editingFinished()
as a result of QCompleter::eventFilter() having forwarded the
return key press event to QComboBox. The second one, would
happen right after, as QCompleter::eventFilter() would process
the same event on behalf of its popup.
(We recall that QCompleter is installed as its own popup event
filter. That's also the case for the completer's widget, although
the purpose there is limited to focus-out events).
The current fix consists on skipping the emit as a result of
QLineEdit::editingFinished() if the completer's popup is still
active. For this to be accurate, it helps to test whether the
completer's popup is visible, so we will not be hiding it in
QWidgetLineControl::processKeyEvent() anymore. Indeed, we know
that if the popup is visible, that means that processKeyEvent()
was called after being forwarded by the completer's popup event
filter. Furthermore, the popup will be hidden by its event filter
shortly after it returns from said event forwarding call.
Based on a patch by Alexey Chernov <4ernov@gmail.com>.
Task-number: QTBUG-51858
Task-number: QTBUG-51889
Change-Id: I013f6c3000ae37b5b0ec20eaf5cf7746c9c903e3
Reviewed-by: Richard Moe Gustavsen <richard.gustavsen@qt.io>
2017-08-28 03:04:35 +00:00
|
|
|
QWidget w;
|
2019-05-17 09:31:00 +00:00
|
|
|
w.setWindowTitle(QLatin1String(QTest::currentTestFunction()));
|
QCompleter: Send activated() signal only once on return key
Due to the complex event forwarding logic between QCompleter,
QComboBox, QLineEdit and QWidgetLineControl, in some cases the
same single user return key press could result in duplicated
activated() signals being emitted by QComboBox. The first one
would be emitted because QLineEdit emitted editingFinished()
as a result of QCompleter::eventFilter() having forwarded the
return key press event to QComboBox. The second one, would
happen right after, as QCompleter::eventFilter() would process
the same event on behalf of its popup.
(We recall that QCompleter is installed as its own popup event
filter. That's also the case for the completer's widget, although
the purpose there is limited to focus-out events).
The current fix consists on skipping the emit as a result of
QLineEdit::editingFinished() if the completer's popup is still
active. For this to be accurate, it helps to test whether the
completer's popup is visible, so we will not be hiding it in
QWidgetLineControl::processKeyEvent() anymore. Indeed, we know
that if the popup is visible, that means that processKeyEvent()
was called after being forwarded by the completer's popup event
filter. Furthermore, the popup will be hidden by its event filter
shortly after it returns from said event forwarding call.
Based on a patch by Alexey Chernov <4ernov@gmail.com>.
Task-number: QTBUG-51858
Task-number: QTBUG-51889
Change-Id: I013f6c3000ae37b5b0ec20eaf5cf7746c9c903e3
Reviewed-by: Richard Moe Gustavsen <richard.gustavsen@qt.io>
2017-08-28 03:04:35 +00:00
|
|
|
w.setLayout(new QVBoxLayout);
|
|
|
|
|
|
|
|
QComboBox cbox;
|
|
|
|
setFrameless(&cbox);
|
|
|
|
cbox.setEditable(true);
|
|
|
|
cbox.setInsertPolicy(QComboBox::NoInsert);
|
2019-05-17 09:31:00 +00:00
|
|
|
cbox.addItems({"foobar1", "foobar2", "bar", "hux"});
|
QCompleter: Send activated() signal only once on return key
Due to the complex event forwarding logic between QCompleter,
QComboBox, QLineEdit and QWidgetLineControl, in some cases the
same single user return key press could result in duplicated
activated() signals being emitted by QComboBox. The first one
would be emitted because QLineEdit emitted editingFinished()
as a result of QCompleter::eventFilter() having forwarded the
return key press event to QComboBox. The second one, would
happen right after, as QCompleter::eventFilter() would process
the same event on behalf of its popup.
(We recall that QCompleter is installed as its own popup event
filter. That's also the case for the completer's widget, although
the purpose there is limited to focus-out events).
The current fix consists on skipping the emit as a result of
QLineEdit::editingFinished() if the completer's popup is still
active. For this to be accurate, it helps to test whether the
completer's popup is visible, so we will not be hiding it in
QWidgetLineControl::processKeyEvent() anymore. Indeed, we know
that if the popup is visible, that means that processKeyEvent()
was called after being forwarded by the completer's popup event
filter. Furthermore, the popup will be hidden by its event filter
shortly after it returns from said event forwarding call.
Based on a patch by Alexey Chernov <4ernov@gmail.com>.
Task-number: QTBUG-51858
Task-number: QTBUG-51889
Change-Id: I013f6c3000ae37b5b0ec20eaf5cf7746c9c903e3
Reviewed-by: Richard Moe Gustavsen <richard.gustavsen@qt.io>
2017-08-28 03:04:35 +00:00
|
|
|
|
|
|
|
cbox.completer()->setCaseSensitivity(Qt::CaseInsensitive);
|
|
|
|
cbox.completer()->setCompletionMode(QCompleter::PopupCompletion);
|
|
|
|
|
|
|
|
w.layout()->addWidget(&cbox);
|
|
|
|
|
2019-05-17 09:31:00 +00:00
|
|
|
w.layout()->addWidget(new QLineEdit);
|
QCompleter: Send activated() signal only once on return key
Due to the complex event forwarding logic between QCompleter,
QComboBox, QLineEdit and QWidgetLineControl, in some cases the
same single user return key press could result in duplicated
activated() signals being emitted by QComboBox. The first one
would be emitted because QLineEdit emitted editingFinished()
as a result of QCompleter::eventFilter() having forwarded the
return key press event to QComboBox. The second one, would
happen right after, as QCompleter::eventFilter() would process
the same event on behalf of its popup.
(We recall that QCompleter is installed as its own popup event
filter. That's also the case for the completer's widget, although
the purpose there is limited to focus-out events).
The current fix consists on skipping the emit as a result of
QLineEdit::editingFinished() if the completer's popup is still
active. For this to be accurate, it helps to test whether the
completer's popup is visible, so we will not be hiding it in
QWidgetLineControl::processKeyEvent() anymore. Indeed, we know
that if the popup is visible, that means that processKeyEvent()
was called after being forwarded by the completer's popup event
filter. Furthermore, the popup will be hidden by its event filter
shortly after it returns from said event forwarding call.
Based on a patch by Alexey Chernov <4ernov@gmail.com>.
Task-number: QTBUG-51858
Task-number: QTBUG-51889
Change-Id: I013f6c3000ae37b5b0ec20eaf5cf7746c9c903e3
Reviewed-by: Richard Moe Gustavsen <richard.gustavsen@qt.io>
2017-08-28 03:04:35 +00:00
|
|
|
|
2019-09-09 14:11:48 +00:00
|
|
|
const auto pos = w.screen()->availableGeometry().topLeft() + QPoint(200,200);
|
QCompleter: Send activated() signal only once on return key
Due to the complex event forwarding logic between QCompleter,
QComboBox, QLineEdit and QWidgetLineControl, in some cases the
same single user return key press could result in duplicated
activated() signals being emitted by QComboBox. The first one
would be emitted because QLineEdit emitted editingFinished()
as a result of QCompleter::eventFilter() having forwarded the
return key press event to QComboBox. The second one, would
happen right after, as QCompleter::eventFilter() would process
the same event on behalf of its popup.
(We recall that QCompleter is installed as its own popup event
filter. That's also the case for the completer's widget, although
the purpose there is limited to focus-out events).
The current fix consists on skipping the emit as a result of
QLineEdit::editingFinished() if the completer's popup is still
active. For this to be accurate, it helps to test whether the
completer's popup is visible, so we will not be hiding it in
QWidgetLineControl::processKeyEvent() anymore. Indeed, we know
that if the popup is visible, that means that processKeyEvent()
was called after being forwarded by the completer's popup event
filter. Furthermore, the popup will be hidden by its event filter
shortly after it returns from said event forwarding call.
Based on a patch by Alexey Chernov <4ernov@gmail.com>.
Task-number: QTBUG-51858
Task-number: QTBUG-51889
Change-Id: I013f6c3000ae37b5b0ec20eaf5cf7746c9c903e3
Reviewed-by: Richard Moe Gustavsen <richard.gustavsen@qt.io>
2017-08-28 03:04:35 +00:00
|
|
|
w.move(pos);
|
|
|
|
w.show();
|
|
|
|
QApplication::setActiveWindow(&w);
|
|
|
|
QVERIFY(QTest::qWaitForWindowActive(&w));
|
|
|
|
|
2020-02-04 14:31:25 +00:00
|
|
|
QSignalSpy activatedSpy(&cbox, &QComboBox::activated);
|
QCompleter: Send activated() signal only once on return key
Due to the complex event forwarding logic between QCompleter,
QComboBox, QLineEdit and QWidgetLineControl, in some cases the
same single user return key press could result in duplicated
activated() signals being emitted by QComboBox. The first one
would be emitted because QLineEdit emitted editingFinished()
as a result of QCompleter::eventFilter() having forwarded the
return key press event to QComboBox. The second one, would
happen right after, as QCompleter::eventFilter() would process
the same event on behalf of its popup.
(We recall that QCompleter is installed as its own popup event
filter. That's also the case for the completer's widget, although
the purpose there is limited to focus-out events).
The current fix consists on skipping the emit as a result of
QLineEdit::editingFinished() if the completer's popup is still
active. For this to be accurate, it helps to test whether the
completer's popup is visible, so we will not be hiding it in
QWidgetLineControl::processKeyEvent() anymore. Indeed, we know
that if the popup is visible, that means that processKeyEvent()
was called after being forwarded by the completer's popup event
filter. Furthermore, the popup will be hidden by its event filter
shortly after it returns from said event forwarding call.
Based on a patch by Alexey Chernov <4ernov@gmail.com>.
Task-number: QTBUG-51858
Task-number: QTBUG-51889
Change-Id: I013f6c3000ae37b5b0ec20eaf5cf7746c9c903e3
Reviewed-by: Richard Moe Gustavsen <richard.gustavsen@qt.io>
2017-08-28 03:04:35 +00:00
|
|
|
|
|
|
|
// Navigate + enter activates only once (first item)
|
|
|
|
cbox.lineEdit()->clear();
|
|
|
|
QTest::keyClick(&cbox, Qt::Key_F);
|
|
|
|
QVERIFY(cbox.completer()->popup());
|
|
|
|
QTRY_VERIFY(cbox.completer()->popup()->isVisible());
|
|
|
|
QTest::keyClick(cbox.completer()->popup(), Qt::Key_Down);
|
|
|
|
QTest::keyClick(cbox.completer()->popup(), Qt::Key_Return);
|
|
|
|
QTRY_COMPARE(activatedSpy.count(), 1);
|
|
|
|
|
|
|
|
// Navigate + enter activates only once (non-first item)
|
|
|
|
cbox.lineEdit()->clear();
|
|
|
|
activatedSpy.clear();
|
|
|
|
QTest::keyClick(&cbox, Qt::Key_H);
|
|
|
|
QVERIFY(cbox.completer()->popup());
|
|
|
|
QTRY_VERIFY(cbox.completer()->popup()->isVisible());
|
|
|
|
QTest::keyClick(cbox.completer()->popup(), Qt::Key_Down);
|
|
|
|
QTest::keyClick(cbox.completer()->popup(), Qt::Key_Return);
|
|
|
|
QTRY_COMPARE(activatedSpy.count(), 1);
|
|
|
|
|
|
|
|
// Full text + enter activates only once
|
|
|
|
cbox.lineEdit()->clear();
|
|
|
|
activatedSpy.clear();
|
|
|
|
QTest::keyClicks(&cbox, "foobar1");
|
|
|
|
QVERIFY(cbox.completer()->popup());
|
|
|
|
QTRY_VERIFY(cbox.completer()->popup()->isVisible());
|
|
|
|
QTest::keyClick(&cbox, Qt::Key_Return);
|
|
|
|
QTRY_COMPARE(activatedSpy.count(), 1);
|
|
|
|
}
|
|
|
|
|
2011-04-27 10:05:43 +00:00
|
|
|
QTEST_MAIN(tst_QCompleter)
|
|
|
|
#include "tst_qcompleter.moc"
|