QT之所以不能再任意控件上绘图,是因为QT的事件过滤器把控件的绘图事件给过滤了。
在paintevent()函数中,通常需要设置QPainter对象,创建QPainter对象的同时需要指定绘图设备,即继承自QPainterDevice的子类作为绘图设备,绘制出来的图形将在这个设备上进行显示,Qt一共提供了4个这样的类,分别是QPixmap,QBitmap,QImage,和QPicture。
在控件上绘图又应该怎么办呢,我这以QLabel为例:使用事件过滤器来使QLabel对象捕获QEvent::Paint事件。即,绘图函数不需要放在paintevent()函数中也可以实现绘图。 实现这个功能的主要函数就还是事件过滤器的两个重要函数,即installEventFilter()和eventFileter() 。
widget.h
#ifndef WIDGET_H
#define WIDGET_H
#include <QWidget>
#include <QPainter>
namespace Ui {
class Widget;
}
class Widget : public QWidget
{
Q_OBJECT
public:
explicit Widget(QWidget *parent = nullptr);
~Widget();
protected:
bool eventFilter(QObject *obj, QEvent *event);
void labelPaint();
private:
Ui::Widget *ui;
};
#endif // WIDGET_H
widget.cpp
#include "widget.h"
#include "ui_widget.h"
Widget::Widget(QWidget *parent) :
QWidget(parent),
ui(new Ui::Widget)
{
ui->setupUi(this);
ui->label->installEventFilter(this); //这行不能省
}
Widget::~Widget()
{
delete ui;
}
bool Widget::eventFilter(QObject *watched, QEvent *event) //用过滤器eventFilter()拦截QLabel中的QEvent::Paint事件
{
if(watched ==ui->label && event->type() == QEvent::Paint)
paint();
return QWidget::eventFilter(watched,event);
}
void Widget::paint() //绘图
{
QPainter painter(ui->label);
painter.setPen(Qt::blue);
// painter.drawLine(100,100,200,200);
painter.drawEllipse(30,15,50,65);
painter.drawLine(0,100,111,100);
}