Как смоделировать действие перетаскивания с помощью QTest - PullRequest
1 голос
/ 10 октября 2019

Чтобы создать тестовый пример для ошибки перетаскивания в виджете QTreeView, я попытался смоделировать поведение при перетаскивании мышью.

Я в основном выбираю первый элемент в QTreeView и хочу, чтобы он перетаскивался на третий элемент. Я сделал это с помощью комбинации QTest::mousePress и QTest::mouseMove. В конце должно быть, конечно, QTest::mouseRelease, но я уже не смог воспроизвести mousePress и mouseMove.

Это шаги, необходимые для воспроизведения:

  1. Пользователь перемещает мышь нацентр первого элемента
  2. пользователь нажимает левую кнопку мыши
  3. пользователь удерживает нажатой левую кнопку мыши и перемещает мышь к центру третьего элемента
  4. пользователь отпускает левую кнопкукнопка мыши

Если выполнить эти действия, как описано, я вижу, что виджеты QTreeView реагируют соответствующим образом и указывают на специальные блики и вертикальные линии, если элемент будет перемещаться между элементами.

К сожалению, мой автоматизированный тест не может воспроизвести это поведение. Кажется, что вызов QTest::mousePress в последовательности делает что-то другое. Также использование пары QTest::mousePress и QTest::mouseMove - это что-то другое.

Это мой код:

main.cpp

#include "TestObject.h"
#include <QTest>

QTEST_MAIN(TestObject)

TestObject.h

#include <QtTest/QtTest>

class TestObject : public QObject
{
    Q_OBJECT

private slots:
    void dragAndDrop();
};

TestObject.cpp

#include "TestObject.h"
#include "TestObject.moc"
#include <QTest>
#include <QTreeView>
#include <QStandardItemModel>
#include <QPropertyAnimation>
#include "MouseMover.h"

void TestObject::dragAndDrop() {
    qDebug() << "Hello";
    QStandardItemModel model;
    QTreeView view;
    view.setModel(&model);
    view.show();
    view.setHeaderHidden(true);
    view.setDragDropMode(QAbstractItemView::DragDropMode::DragDrop);
    view.setDefaultDropAction(Qt::DropAction::MoveAction);
    view.setColumnHidden(1, true);
    for (auto rowIter = 0; rowIter < 3; rowIter++) {
        QList<QStandardItem*> items;
        for (auto colIter = 0; colIter < 2; colIter++) {
            items << new QStandardItem(QString("%1-%2").arg(rowIter).arg(colIter));
        }
        model.appendRow(items);
    }

    MouseMover mover;
    mover.setWidget(view.viewport());
    QPropertyAnimation anim(&mover, "mousePosition");

    QTimer::singleShot(0, [&]() {
        auto startValue = view.visualRect(model.index(0, 0)).center();
        auto endValue = view.visualRect(model.index(2, 0)).center();

        QTest::mousePress(view.viewport(), Qt::MouseButton::LeftButton, Qt::KeyboardModifier::NoModifier, startValue);
        anim.setStartValue(startValue);
        anim.setEndValue(endValue);
        anim.setDuration(500);
        anim.start();
    });
    qApp->exec();
}

MouseMover.h

#pragma once
#include <QObject>
#include <QTest>

class MouseMover : public QObject {
Q_OBJECT
public:
    Q_PROPERTY(QPoint mousePosition READ mousePosition WRITE setMousePosition MEMBER mMousePosition)

    void setWidget(QWidget* widget) {
        mWidget = widget;
    }
    QPoint mousePosition() const {
        return mMousePosition;
    }

    void setMousePosition(const QPoint& pos) {
        mMousePosition = pos;
        if (mWidget) {
            QTest::mousePress(mWidget, Qt::MouseButton::LeftButton, Qt::KeyboardModifier::NoModifier, mMousePosition);
            QTest::mouseMove(mWidget, mMousePosition);

        }
    }

private:
    QPoint mMousePosition;
    QWidget* mWidget{ nullptr };
};

MouseMover.cpp

#include "MouseMover.h"
...