如何从QTest单元测试中编辑QTableView单元?

时间:2012-09-26 15:03:11

标签: qt qt4 qtableview qt4.6 qtestlib

我正在使用QTestLib framework在QTableView中为自定义Validator编写单元测试。其中一个最基本的测试用例可以这样描述:

  

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

仅仅更改模型中的值或任何内容是不够的,测试用例应该像这样执行:

  1. 双击表格单元格将其设置为编辑模式
  2. 按[结束]键。
  3. 按[5]键。
  4. 注意:This question有一个关于如何从代码中将表格单元格设置为编辑模式的答案,但是单元测试应该尝试坚持人类用户的可能性,即鼠标/键盘操作。

    我发现可以使用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下尝试这个

1 个答案:

答案 0 :(得分:7)

尝试通过模拟事件在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)命令)


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