QTableWidget中的表格显示图片有几种方式:
需要先将图片加载为资源。
1.直接在创建QTableWidgetItem的时候创建,如
tableWdiget->setItem(0,1,new QTableWidgetItem(QIcon(":/image/cpu")," "));
2.通过在item中创建一个label,通过label的setPixMap函数来显示图片:
QLabel *label = new QLabel("");
label->setPixmap(QPixmap(":/image/grid").scaled(30,30));
ui->matrixViewTable->setCellWidget(0,i,label);
3.通过继承 QStyledItemDelegate类使用委托来实现item中显示图片:
class QPixmapItemdele : public QStyledItemDelegate
{
public:
QPixmapItemdele(QObject* parent = 0):QStyledItemDelegate(parent){ }
//在委托类的paint中画图
virtual void paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const
{
if(index.data(Qt::DisplayRole).canConvert<QPixmap>())
{
QPixmap pix = index.data(Qt::DisplayRole).value<QPixmap>();
painter->drawPixmap(option.rect,pix);
}
QStyledItemDelegate::paint(painter,option,index);
}
};然后为tablewidget设置委托,后面就可以设置具体的某项的图片:
ui->matrixViewTable->setItemDelegate(new QPixmapItemdele());
//以下是设置第一行的所有列的图片
for(int i=0; i<ui->matrixViewTable->columnCount(); i++)
{
QTableWidgetItem *item = new QTableWidgetItem();
ui->matrixViewTable->setItem(0,i,item);
item->setData(Qt::DisplayRole,QVariant::fromValue(QPixmap(":/image/grid").scaled(30,30)));
}
以上三种方法,法一直接在item上创建Icon,则表中每个Item表示为左边icon,右边为icon对应的text,即使text为空,也会留出很大的空间出来,导致左边的icon很小,
可以用以下语句来设置左边图标大小:
ui->matrixViewTable->setIconSize(QSize(30,30));
但是右边还是有text留出来的空间,如果想让icon充满整个item,则此方法没法达成。
法二通过item中创建label来setPixMap的方式可以让icon充满整个item,但是显示出来的icon用鼠标点击不会有选择动作发生,或者也许可以处理点击事件(未试过,不知是否能行)。
法三继承委托能够让图片充满整个item,同时选择图片时,会出现选择框,不至于像法二一样点击了没有任何反应。
以下是法三QWidgetTable使用委托的代码(委托类已在上面给出):
ui->matrixViewTable->setItemDelegate(new QPixmapItemdele());
int colcount = 17;
int rowcount = 2;
ui->matrixViewTable->setColumnCount(colcount);
ui->matrixViewTable->setRowCount(rowcount);
ui->matrixViewTable->horizontalHeader()->setVisible(false);
ui->matrixViewTable->verticalHeader()->setVisible(false);
ui->matrixViewTable->setSelectionMode(ui->matrixViewTable->NoSelection);
ui->matrixViewTable->setEditTriggers(ui->matrixViewTable->SelectedClicked);
ui->matrixViewTable->setShowGrid(false);
ui->matrixViewTable->setFrameShape(QFrame::NoFrame);
const int width = 30;
for(int i=0; i<ui->matrixViewTable->columnCount(); i++)
{
ui->matrixViewTable->setColumnWidth(i,width);
//设置第8列为间隔行,不允许点击或选择
if(i==8)
{
QTableWidgetItem *itm1 = new QTableWidgetItem();
ui->matrixViewTable->setItem(0,i,itm1);
itm1->setFlags(Qt::NoItemFlags);
QTableWidgetItem *itm2 = new QTableWidgetItem();
ui->matrixViewTable->setItem(1,i,itm2);
itm2->setFlags(Qt::NoItemFlags);
continue;
}
QTableWidgetItem *item = new QTableWidgetItem();
ui->matrixViewTable->setItem(0,i,item);
item->setData(Qt::DisplayRole,QVariant::fromValue(QPixmap(":/image/grid").scaled(30,30)));
QTableWidgetItem *item2 = new QTableWidgetItem();
ui->matrixViewTable->setItem(1,i,item2);
item2->setData(Qt::DisplayRole,QVariant::fromValue(QPixmap(":/image/grid").scaled(30,30)));
}
ui->matrixViewTable->setColumnWidth(8,15);以下是运行生成的效果,每个表格中间是一个图片:
使用委托能够在表格中显示图片,但是如果要在表格中画图,比如画线,则使用委托画出来的图会把线覆盖掉,也就是线会有画出来,但是看不到,实际是画了在图的下面。
在表格中显示图片还有第4种方法,安装事件过滤器,然后在过滤器函数EventFilter()函数中进行画图。因为此处用到的QTableWidget是Widget中的一个控件,因此可以安装事件过滤器,让Widget监视QTableWidget中的事件。代码如下,在类中重写QWidget类的eventFilter函数,然后创建表格时,每一列就创建为空,在evenFilter函数中检测到paint事件时才进行将图片画到表格中。
class MatrixView : public QWidget
{
Q_OBJECT
public:
explicit MatrixView(QWidget *parent = 0);
~MatrixView();
public:
bool eventFilter(QObject *, QEvent *);
private:
void InitTableWidget();
bool IsMouseInTableWidget(QEvent *event, QPoint &p);
void SetPointToCenter(QPoint&);
private:
Ui::MatrixView *ui;
QPoint m_beginP;
QPoint m_endP;
bool m_bmove;
bool m_brelease;
};//以下是在构造函数中创建表格,表格其他部分与上面代码一样
m_bmove = false;
m_brelease = false;
//必须是tablewidget的viewport注册事件过滤才会有鼠标事件,如果是tablewidget注册,则收到除了鼠标事件外的其他事件,也就是鼠标事件收不到
ui->matrixViewTable->viewport()->installEventFilter(this);
for(int i=0; i<ui->matrixViewTable->columnCount(); i++)
{
ui->matrixViewTable->setColumnWidth(i,COL_WIDTH);
//设置第8列为间隔行,不允许点击或选择
if(i == NULL_COL)
{
QTableWidgetItem *itm1 = new QTableWidgetItem();
ui->matrixViewTable->setItem(0,i,itm1);
itm1->setFlags(Qt::NoItemFlags);
QTableWidgetItem *itm2 = new QTableWidgetItem();
ui->matrixViewTable->setItem(1,i,itm2);
itm2->setFlags(Qt::NoItemFlags);
continue;
}
QTableWidgetItem *item = new QTableWidgetItem();
ui->matrixViewTable->setItem(0,i,item);
//使用委托显示图片
//item->setData(Qt::DisplayRole,QVariant::fromValue(QPixmap(pixmap).scaled(30,30)));
QTableWidgetItem *item2 = new QTableWidgetItem();
ui->matrixViewTable->setItem(1,i,item2);
//item2->setData(Qt::DisplayRole,QVariant::fromValue(QPixmap(pixmap).scaled(30,30)));
}
//在eventFilter中获取tablewidget的鼠标按下、移动,释放事件,进行画线,在画线之前进行画图。此处画图的效果与法三的效果一样,且可以在图上面画线。如果不需要处理鼠标事件,此处可以忽略3个鼠标事件的判断。
bool MatrixView::eventFilter(QObject *object, QEvent *event)
{
// qDebug("%s type:%d",object->objectName().toStdString().c_str(),event->type());
if(object->objectName()!=QString("qt_scrollarea_viewport"))
{
return QWidget::eventFilter(object,event);
}
if(event->type() == QEvent::MouseButtonPress)
{
if(IsMouseInTableWidget(event,m_beginP))
{
m_endP = m_beginP;
update();
}
}
if(event->type() == QEvent::MouseButtonRelease)
{
IsMouseInTableWidget(event,m_endP);
m_brelease = true;
SetPointToCenter(m_endP);
update();
}
if(event->type() == QEvent::MouseMove)
{
QMouseEvent *mouseEvent = static_cast<QMouseEvent*>(event);
//不进行判断左右键,如果判断左右键,会导致m_endP有时无值,画出的线断断续续
m_endP = mouseEvent->pos();
m_bmove = true;
update();
}
if(event->type() == QEvent::Paint)
{
QPainter paint(ui->matrixViewTable->viewport());
//使用paint事件在表格中画图
int w = COL_WIDTH;
int h = ui->matrixViewTable->rowHeight(0);
for(int i=0; i<ui->matrixViewTable->columnCount()-1; i++)
{
int x = i>=NULL_COL ? i*COL_WIDTH+NULL_COL_WIDTH : i*COL_WIDTH;
int y = 0;
int y1 = ui->matrixViewTable->rowHeight(0);
QPixmap pix(":/image/grid");
paint.drawPixmap(QRect(x,y,w,h),pix);
QPixmap pix1(":/image/grid");
paint.drawPixmap(QRect(x,y1,w,h),pix1);
}
//在表格中画线,线随鼠标动
if(m_bmove)
{
QPen pen(Qt::black,5);
paint.setPen(pen);
SetPointToCenter(m_beginP); //取得鼠标最接近项的中心点
paint.drawPoint(m_beginP);
QPen pen1(Qt::black,3);
paint.setPen(pen1);
paint.drawLine(m_beginP,m_endP);
QPen pen2(Qt::black,5);
paint.setPen(pen2);
paint.drawPoint(m_endP);
}
if(m_brelease)
{
m_bmove = false;
m_brelease = false;
}
}
//此处必须返回基类的eventFilter,以让基类有机会进行处理其他事情,否则单单返回true,Qtablewidget上面将无其他东西显示
return QWidget::eventFilter(object,event);
}
//判断鼠标按下的点是否在表格内,可无需关注
bool MatrixView::IsMouseInTableWidget(QEvent *event, QPoint &point)
{
if(event->type() == QEvent::MouseButtonPress ||
event->type() == QEvent::MouseButtonRelease||
event->type() == QEvent::MouseMove)
{
QMouseEvent *mouseEvent = static_cast<QMouseEvent*>(event);
if(mouseEvent->button() == Qt::LeftButton)
{
QPoint p = mouseEvent->pos();
//检查鼠标位置是否点在列表的项中,需要考虑中间空白间隔的那列除外
int col = p.x()/COL_WIDTH>=NULL_COL ? (p.x()+NULL_COL_WIDTH)/COL_WIDTH : p.x()/COL_WIDTH;
int row = p.y()/ui->matrixViewTable->rowHeight(0);
if(col<ui->matrixViewTable->columnCount() && row<ui->matrixViewTable->rowCount())
{
if(col != NULL_COL)
{
point = p;
return true;
}
}
}
}
return false;
}
//取得最靠近该点的项的中心点,可无需关注
void MatrixView::SetPointToCenter(QPoint &point)
{
int x = 0;
if(point.x()/COL_WIDTH < NULL_COL)
{
x = point.x()/COL_WIDTH*COL_WIDTH + COL_WIDTH/2;
}
else
{
//因为第8列是空列,所以要排除这一列,先将point所在的x坐标减去8列前的部分,得到超出8列后的部分,就可以按照取中心点的方法取得这超出后部分的中心点,再加上8列前部分的位置,就得到最终的中心点
int null_col_end_x = NULL_COL*COL_WIDTH + NULL_COL_WIDTH;
x = null_col_end_x + (point.x()-null_col_end_x)/COL_WIDTH*COL_WIDTH + COL_WIDTH/2;
}
int h = ui->matrixViewTable->rowHeight(0);
int y = point.y()/h*h + h/2;
point = QPoint(x,y);
}效果 图:(图中直线上的折线是抗锯齿,如果线宽小的话就不会这么明显)