Qt 重写 QLabel PaintEvent

Posted

技术标签:

【中文标题】Qt 重写 QLabel PaintEvent【英文标题】:Qt Overriding QLabel PaintEvent 【发布时间】:2017-03-03 05:53:17 【问题描述】:

过去几天我一直在努力解决这个问题。当用户调整窗口大小时,我希望能够扩大和缩小 QLabel 中分配的像素图。问题是保持纵横比和图像质量。这里的另一位用户建议我重新实现标签的绘制事件 - 但我仍然很迷茫。我什至不确定我是否正确覆盖了paintEvent。我会在这里杀死一些示例代码。

这就是我所在的地方:

void MyLabel::paintEvent(QPaintEvent * event)

    //if this widget is assigned a pixmap
    //paint that pixmap at the size of the parent, aspect ratio preserved
    //otherwise, nothing

【问题讨论】:

你看过related question中的solution吗?它看起来非常接近您想要实现的目标。 【参考方案1】:

这是一个QLabel 子类的可能实现,它在保持纵横比的同时缩放其像素图内容。基于QLabel::paintEventis implemented的方式实现。

如果您在布局中使用它,您还可以将其大小策略设置为QSizePolicy::Expanding,以便QLabel 占用布局中的额外空间,以便更大地显示像素图内容。

#include <QApplication>
#include <QtWidgets>

class PixmapLabel : public QLabel
public:
    explicit PixmapLabel(QWidget* parent=nullptr):QLabel(parent)
        //By default, this class scales the pixmap according to the label's size
        setScaledContents(true);
    
    ~PixmapLabel()

protected:
    void paintEvent(QPaintEvent* event);
private:
    //QImage to cache the pixmap()
    //to avoid constructing a new QImage on every scale operation
    QImage cachedImage;
    //used to cache the last scaled pixmap
    //to avoid calling scale again when the size is still at the same
    QPixmap scaledPixmap;
    //key for the currently cached QImage and QPixmap
    //used to make sure the label was not set to another QPixmap
    qint64 cacheKey0;
;

//based on the implementation of QLabel::paintEvent
void PixmapLabel::paintEvent(QPaintEvent *event)
    //if this is assigned to a pixmap
    if(pixmap() && !pixmap()->isNull())
        QStyle* style= PixmapLabel::style();
        QPainter painter(this);
        drawFrame(&painter);
        QRect cr = contentsRect();
        cr.adjust(margin(), margin(), -margin(), -margin());
        int align= QStyle::visualAlignment(layoutDirection(), alignment());
        QPixmap pix;
        if(hasScaledContents()) //if scaling is enabled
            QSize scaledSize= cr.size() * devicePixelRatioF();
            //if scaledPixmap is invalid
            if(scaledPixmap.isNull() || scaledPixmap.size()!=scaledSize
                    || pixmap()->cacheKey()!=cacheKey)
                //if cachedImage is also invalid
                if(pixmap()->cacheKey() != cacheKey)
                    //reconstruct cachedImage
                    cachedImage= pixmap()->toImage();
                
                QImage scaledImage= cachedImage.scaled(
                            scaledSize, Qt::KeepAspectRatio, Qt::SmoothTransformation);
                scaledPixmap= QPixmap::fromImage(scaledImage);
                scaledPixmap.setDevicePixelRatio(devicePixelRatioF());
            
            pix= scaledPixmap;
         else  // no scaling, Just use pixmap()
            pix= *pixmap();
        
        QStyleOption opt;
        opt.initFrom(this);
        if(!isEnabled())
            pix= style->generatedIconPixmap(QIcon::Disabled, pix, &opt);
        style->drawItemPixmap(&painter, cr, align, pix);
     else  //otherwise (if the label is not assigned to a pixmap)
        //call base paintEvent
        QLabel::paintEvent(event);
    


//DEMO program
QPixmap generatePixmap(QSize size) 
    QPixmap pixmap(size);
    pixmap.fill(Qt::white);
    QPainter p(&pixmap);
    p.setRenderHint(QPainter::Antialiasing);
    p.setPen(QPen(Qt::black, 10));
    p.drawEllipse(pixmap.rect());
    p.setPen(QPen(Qt::red, 2));
    p.drawLine(pixmap.rect().topLeft(), pixmap.rect().bottomRight());
    p.drawLine(pixmap.rect().topRight(), pixmap.rect().bottomLeft());
    return pixmap;



int main(int argc, char *argv[])


    QApplication a(argc, argv);


    QPixmap pixmap= generatePixmap(QSize(1280, 960));
    PixmapLabel label;
    label.setPixmap(pixmap);
    label.setAlignment(Qt::AlignCenter);
    label.setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
    label.setMinimumSize(320, 240);
    label.show();

    return a.exec();

我认为这比this answer 中的解决方案要好,因为这里的QLabel 负责调整其像素图的大小。因此,无需在每次调整父窗口小部件的大小以及在其上设置新的像素图时手动调整其大小。

【讨论】:

感谢您的帮助,这一切都清楚了。【参考方案2】:

首先我要感谢 Mike。

此外,我想在 2016 年 10 月 21 日对他的回答进行补充 - 但是我目前还无法添加评论 - 因此这里有一个完整的答案。 [如果有人能够将其移至评论部分,请随意。]

我还添加了对 QLabel 的其他构造函数和窗口标志的覆盖 [使用 QLabel 中的默认参数] 并添加了 Q_OBJECT 宏,以便与 Qt 框架兼容:

标题:

class PixmapLabel : public QLabel

    Q_OBJECT
public:
    explicit PixmapLabel(QWidget* parent=nullptr, Qt::WindowFlags f = Qt::WindowFlags());
    explicit PixmapLabel(const QString &text, QWidget *parent = nullptr, Qt::WindowFlags f = Qt::WindowFlags());
   ...

模块:

PixmapLabel::PixmapLabel(QString const & text, QWidget * parent, Qt::WindowFlags f) :
    QLabel(text, parent, f)

    //By default, this class scales the pixmap according to the label's size
     setScaledContents(true);

【讨论】:

以上是关于Qt 重写 QLabel PaintEvent的主要内容,如果未能解决你的问题,请参考以下文章

QT5中怎样connect QPushButton和QLabel

Qt之在控件上绘图

QWidget QLabel没有响应keyEvent键盘事件

QWidget QLabel没有响应keyEvent键盘事件

新手:Qt之QLabel类的应用

Qt qss问题总结