这是我在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将它们发送到我的应用程序。到目前为止,这种方法运作良好。现在我需要从我的应用程序中更改这些选择,这就是问题所在。我的首选方法是使用鼠标点击进行操作。这背后的意图是,它表现得好像用户正在该应用程序中工作(但似乎并非如此)。 但如果你有另一个想法,我会尝试一下。也许有更好的东西,但我还没想到。
答案 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)。