QT:使用QMouseEvent从代码中选择多个QTableWidgetItems

时间:2014-05-15 17:48:39

标签: qt qtablewidget qtablewidgetitem qmouseevent

这是我在stackoverflow上的第一个问题,但不是你第一次帮我(我希望你能)。

首先我要提到的是,我不熟悉C ++,尤其是QT,我的问题就在于此。 我在将QMouseEvents从代码发布(或发送)到QTableWidget时遇到问题。 我想向MultiSelect QTableWidget发送多次点击,我希望,之后会选择点击的QTableWidgetItem。

为了重现这个问题,我创建了一个小样本。有一个按钮可以发送"点击"到QTableWidget(在我的真实应用程序中,这是由网络触发)。 单击按钮时,两次"单击"执行(我也在调试器中看到),但之后只选择了第一个单击的TableWidgetItem。 我无法弄清楚为什么第二个项目没有被选中。我希望有人可以:-)。我真的很感谢你的帮助!

示例代码:

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    QTableWidgetExamples w;
    w.show();
    return a.exec();
}

QTableWidgetExample.h:

#ifndef QTABLEWIDGETEXAMPLES_H
#define QTABLEWIDGETEXAMPLES_H

#include <QtGui/QMainWindow>
#include "ui_qtablewidgetexamples.h"

QT_BEGIN_NAMESPACE  // QT_BEGIN_NAMESPACE / QT_END_NAMESPACE are not needed in Qt user code
class QTableWidget; //forward declaration
class QTableWidgetItem; //forward declaration
QT_END_NAMESPACE

class QTableWidgetExamples : public QMainWindow
{
    Q_OBJECT

public:
    QTableWidgetExamples(QWidget *parent = 0, Qt::WFlags flags = 0);
    ~QTableWidgetExamples();

private:
    Ui::QTableWidgetExamplesClass _ui;
    QTableWidget* _tableWidget;
    void clickItemFromCode(int index);

public slots:
    void btnClicked();
    void onItemClick(QTableWidgetItem* item);
};

#endif // QTABLEWIDGETEXAMPLES_H

QTableWidgetExample.cpp:

#include <QTableWidget>
#include <QPushButton>
#include <QVBoxLayout>
#include <QMouseEvent>
//#include <QTest>
#include "qtablewidgetexamples.h"

QTableWidgetExamples::QTableWidgetExamples(QWidget *parent, Qt::WFlags flags)
    : QMainWindow(parent, flags)
{
    _ui.setupUi(this);
    QVBoxLayout *vlay = new QVBoxLayout(this);

    _tableWidget = new QTableWidget(3,1,this);
    _tableWidget->verticalHeader()->hide();
    _tableWidget->horizontalHeader()->hide();
    _tableWidget->setSelectionMode(QAbstractItemView::SelectionMode::MultiSelection);
    _tableWidget->setObjectName("list_1");
    _tableWidget->setItem(0, 0, new QTableWidgetItem("Item_0"));
    _tableWidget->setItem(1, 0, new QTableWidgetItem("Item_1"));
    _tableWidget->setItem(2, 0, new QTableWidgetItem("Item_2"));
    connect(_tableWidget, SIGNAL(itemClicked(QTableWidgetItem*)), this, SLOT(onItemClick(QTableWidgetItem*)));

    QPushButton *btn1 = new QPushButton("btn1");
    connect(btn1, SIGNAL(clicked()), this, SLOT(btnClicked()));

    vlay->addWidget(_tableWidget);
    vlay->addWidget(btn1);

    QWidget * wdg = new QWidget(this);
    wdg->setLayout(vlay);
    setCentralWidget(wdg);
}

QTableWidgetExamples::~QTableWidgetExamples() {}

void QTableWidgetExamples::onItemClick(QTableWidgetItem *item)
{
    bool isSel = item->isSelected();
    QString text = item->text();
}

void QTableWidgetExamples::btnClicked()
{
    clickItemFromCode(1);
    clickItemFromCode(2);   
}

void QTableWidgetExamples::clickItemFromCode(int rowIndex)
{
    float subX, subY;
    subX = subY = 0.5;

    QModelIndex index = _tableWidget->model()->index(rowIndex, 0);
    QRect rc = _tableWidget->visualRect(index); // view()->visualRect(index);
    QPoint pt(rc.x()+rc.width()*subX,rc.y()+rc.height()*subY);

    // just for debug purposes
    QTableWidgetItem* item = _tableWidget->item(rowIndex,0);
    QString itemText = item->text();

    QWidget *widget = (QWidget*) _tableWidget; //view();
    widget = widget->childAt(10,10);

    QEvent *e = QMouseEvent::createExtendedMouseEvent(QEvent::MouseMove,
            pt,widget->mapToGlobal(pt), Qt::LeftButton,Qt::LeftButton,0);
    QApplication::sendEvent(widget, e);

    e = QMouseEvent::createExtendedMouseEvent(QEvent::MouseButtonPress,
            pt,widget->mapToGlobal(pt), Qt::LeftButton,Qt::LeftButton,0);
    QApplication::sendEvent(widget, e);

    e = QMouseEvent::createExtendedMouseEvent(QEvent::MouseButtonRelease,
            pt,widget->mapToGlobal(pt), Qt::LeftButton,Qt::LeftButton,0);
    QApplication::sendEvent(widget, e);

    //QApplication::sendPostedEvents();
}

编辑为mbroadst回答:

不幸的是,这是一项艰难的要求。我将我的代码注入另一个应用程序,在那里我读了几个QTableWidgets并通过TCP将它们发送到我的应用程序。到目前为止,这种方法运作良好。现在我需要从我的应用程序中更改这些选择,这就是问题所在。我的首选方法是使用鼠标点击进行操作。这背后的意图是,它表现得好像用户正在该应用程序中工作(但似乎并非如此)。 但如果你有另一个想法,我会尝试一下。也许有更好的东西,但我还没想到。

2 个答案:

答案 0 :(得分:0)

您可以使用QTableWidgetItem::setSelected()以编程方式选择项目。所以你的代码看起来像是:

void QTableWidgetExamples::clickItemFromCode(int rowIndex)
{
    QTableWidgetItem *item = tableWidget->item(rowIndex, 0);
    item->setSelected(true);
}

或者,如果您真的需要模拟鼠标点击,那么:

// NOTE: I haven't explicitly tested this, but its the general idea
void QTableWidgetExamples::clickItemFromCode(int rowIndex)
{
    QTableWidgetItem *item = tableWidget->item(rowIndex, 0);
    QRect visualItemRect = tableWidget->visualRectItem(item);

    QMouseEvent *event = new QMouseEvent(QEvent::MouseButtonRelease, visualItemRect.center(),     
                                         Qt::LeftButton, Qt::LeftButton, Qt::NoModifier);
    QCoreApplication::sendEvent(tableWidget, event);
}

答案 1 :(得分:0)

最后我得到了一个有效的解决方案:

QEvent::MouseMove未正确设置。

这是更正的。在交换之后,示例代码完成它应该做的事情。

QEvent *e = QMouseEvent::createExtendedMouseEvent(QEvent::MouseMove,
            pt,widget->mapToGlobal(pt), Qt::NoButton, Qt::RightButton, Qt::NoModifier);
QApplication::postEvent(widget, e);

如果您不需要模拟鼠标点击,@mbroadst的第一种方法(以编程方式设置所选项目)可能是更好的选择(thx for that)。