9

我正在使用QTestLib 框架为 QTableView 中的自定义验证器编写单元测试。最基本的测试用例之一可以这样描述:

双击第三列第四行的表格单元格,并将数字“5”附加到其内容中。

仅仅改变模型或任何东西中的值是不够的,测试用例应该像这样执行它:

  1. 双击表格单元格将其设置为编辑模式
  2. 按[结束]键。
  3. 按 [5] 键。

注意:这个问题有一个关于如何从代码中将表格单元格设置为编辑模式的答案,但是单元测试应尽量坚持人类用户的可能性,即鼠标/键盘操作。

我发现可以使用QTableView::columnViewportPosition( int )QTableView::rowViewportPosition( int )检索单元格的 X/Y 位置。但是,使用QTest::mouseDClick(...)在指定位置双击既不会选择单元格,也不会将其设置为编辑模式:

// Retrieve X/Y coordinates of the cell in the third column and the fourth row
int xPos = m_pTableView->columnViewportPosition( 2 );
int yPos = m_pTableView->rowViewportPosition( 3 );

// This does not work
QTest::mouseDClick( m_pTableView, Qt::LeftButton, QPoint( xPos, yPos ) );

如何仅使用鼠标/键盘操作来实现我上面描述的测试用例?

PS:我在 Windows XP 32 位和 Qt 4.6.1 下尝试这个

4

1 回答 1

11

尝试通过模拟事件在 QTableView 中进行编辑时需要考虑几件事:

QTableView 不直接显示其单元格,它使用它的viewport()来显示。同样,双击事件必须发送到视口而不是表视图本身。

现在当你做

QTest::mouseDClick( m_pTableView->viewport(), Qt::LeftButton, 
                    NULL, QPoint( xPos, yPos ) );

单元格将被选中,但不会处于编辑模式(与人为启动的双击不同,即使表格视图之前没有焦点,也会立即将单元格置于编辑模式)。但是,如果您在双击之前在同一位置单击一次,它将起作用!

如果您随后将 [End] 按键发送到视口,则光标不会跳到表格单元格内容的末尾,而是会选择当前行中的最后一个单元格。
为了更改表格单元格内容,您必须改为将事件发送到当前编辑器小部件。最简单的方法是使用QWidget::focusWidget()

QTest::keyClick( m_pTableView->viewport()->focusWidget(), Qt::Key_End );

请注意,尽管 focusWidget() 可以返回 NULL,但这样使用它可能是不安全的。


有了这些知识,测试用例可以编写如下:

// Note: The table view must be visible at this point

// Retrieve X/Y coordinates of the cell in the third column and the fourth row
int xPos = m_pTableView->columnViewportPosition( 2 ) + 5;
int yPos = m_pTableView->rowViewportPosition( 3 ) + 10;

// Retrieve the viewport of the table view
QWidget* pViewport = m_pTableView->viewport();

// Double click the table cell to set it into editor mode
// Note: A simple double click did not work, Click->Double Click works, however
QTest::mouseClick ( pViewport, Qt::LeftButton, NULL, QPoint( xPos, yPos ) );
QTest::mouseDClick( pViewport, Qt::LeftButton, NULL, QPoint( xPos, yPos ) );

// Simulate [End] keypress
QTest::keyClick( pViewport->focusWidget(), Qt::Key_End );

// Simulate [5] keypress
QTest::keyClick( pViewport->focusWidget(), Qt::Key_5 );

(注意:如果要验证这一点,可以在每个事件后添加 QTest::qWait(1000) 命令)


如果您使用此处所述的 _data() 函数,请注意在创建数据时无法检索 focusWidget()。我通过创建一个ITestAction只有纯虚拟“execute()”函数的自定义接口解决了这个问题。然后,我添加了具有与函数类似的构造QTest::mouseClick(...) etc函数的子类。这些类只调用 QTest 函数,但使用小部件本身或其焦点小部件作为参数,这取决于附加的布尔标志。_data() 槽然后为每个数据行存储一个 QList< ITestAction* >,实际的测试槽迭代该列表并在执行验证之前对每个项目调用 execute()。

于 2012-09-26T15:03:11.407 回答