Qt-继承自QLabel的图像缩放移动类

文章描述了一个使用Qt开发的ImageTransformLabel类,它扩展了QLabel,提供了图像加载、放大/缩小、右键菜单和图像位置调整功能。用户可以通过鼠标操作改变图像视图并执行预设操作。
摘要由CSDN通过智能技术生成

头文件如下:

#ifndef IMAGETRANSFORMLABEL_H
#define IMAGETRANSFORMLABEL_H

#include <QLabel>
#include <QMenu>
#include <QContextMenuEvent>
#include <QWheelEvent>
#include <QMouseEvent>
#include <QPaintEvent>
#include <QCloseEvent>

class ImageTransformLabel : public QLabel
{
    Q_OBJECT
public:
    explicit ImageTransformLabel(QWidget *parent = nullptr);
    void initMenu();
    void setImage(QString imagePath);

protected:

    void contextMenuEvent(QContextMenuEvent* event) override;
    void paintEvent(QPaintEvent* event);
    void wheelEvent(QWheelEvent* event) override;
    void mousePressEvent(QMouseEvent* event) override;
    void mouseMoveEvent(QMouseEvent* event) override;
    void mouseReleaseEvent(QMouseEvent* event) override;

private:
    void slot_ActionLoadImage();
    void slot_ActionZoomInImage();
    void slot_ActionZoomOutImage();
    void slot_ActionPresetImage();
    void slot_ActionClearImage();

private:
    QImage mImage;//加载的图像图像
    QString mImagePath;//图像路径
    float mZoomValue;//放大系数
    float mOffsetX;//X方向偏移值
    float mOffsetY;//Y方向偏移值

    QPoint mOldPos;
    bool mPressedFlag;

    QMenu* mMenu;//右键菜单
    QAction* mActionLoadImage;//加载图像
    QAction* mActionZoomInImage;//放大图像
    QAction* mActionZoomOutImage;//缩小图像
    QAction* mActionPresetImage;//还原图像
    QAction* mActionClearImage;//清空图像

signals:

};

#endif // IMAGETRANSFORMLABEL_H

源文件如下:

#include "imagetransformlabel.h"
#include <QFileDialog>
#include <QPainter>
#include <QMessageBox>

#pragma execution_character_set("utf-8")

ImageTransformLabel::ImageTransformLabel(QWidget *parent) : QLabel(parent)
{
    initMenu();//初始化菜单
}

//初始化右键菜单
void ImageTransformLabel::initMenu()
{
    mMenu = new QMenu(this);

    mActionLoadImage = new QAction();
    mActionLoadImage->setText("加载图像");
    mMenu->addAction(mActionLoadImage);
    mMenu->addSeparator();//添加分割符

    mActionZoomInImage = new QAction();
    mActionZoomInImage->setText("放大图像");
    mMenu->addAction(mActionZoomInImage);
    mMenu->addSeparator();//添加分割符

    mActionZoomOutImage = new QAction();
    mActionZoomOutImage->setText("缩小图像");
    mMenu->addAction(mActionZoomOutImage);
    mMenu->addSeparator();//添加分割符

    mActionPresetImage = new QAction();
    mActionPresetImage->setText("还原图像");
    mMenu->addAction(mActionPresetImage);
    mMenu->addSeparator();//添加分割符

    mActionClearImage = new QAction();
    mActionClearImage->setText("清除图像");
    mMenu->addAction(mActionClearImage);

    connect(mActionLoadImage,&QAction::triggered,this,&ImageTransformLabel::slot_ActionLoadImage);
    connect(mActionZoomInImage,&QAction::triggered,this,&ImageTransformLabel::slot_ActionZoomInImage);
    connect(mActionZoomOutImage,&QAction::triggered,this,&ImageTransformLabel::slot_ActionZoomOutImage);
    connect(mActionPresetImage,&QAction::triggered,this,&ImageTransformLabel::slot_ActionPresetImage);
    connect(mActionClearImage,&QAction::triggered,this,&ImageTransformLabel::slot_ActionClearImage);
}

void ImageTransformLabel::setImage(QString imagePath)
{
    if(imagePath.isEmpty())
        return;

    mZoomValue = 1.0;
    mOffsetX = 0;
    mOffsetY = 0;
    mImagePath = imagePath;
    mImage.load(mImagePath);//加载图像
    update();
}

void ImageTransformLabel::contextMenuEvent(QContextMenuEvent* event)
{
    if(!mMenu)
        return;
    mMenu->exec(QCursor::pos());

}


void ImageTransformLabel::paintEvent(QPaintEvent *event)
{   
    if(mImage.isNull())//mImage是QImage类图像
        return QWidget::paintEvent(event);

    QPainter painter(this);


    //根据窗口计算应该显示的图像大小-算法留意下
    int width = qMin(mImage.width(),this->width());//qMin函数返回两个参数的最小值
    int height = int(width * 1.0 /(mImage.width() * 1.0 / mImage.height()));
    height = qMin(height,this->height());
    width = height * 1.0 * (mImage.width() * 1.0 / mImage.height());

    //平移
    painter.translate(this->width()/2 + mOffsetX,this->height()/2 + mOffsetY);//mOffsetX是x方向偏移值,mOffsetY是y方向偏移值

    //缩放
    painter.scale(mZoomValue,mZoomValue);//mZoomValue是缩放系数

    //绘制图像
    QRect picRect(-width/2,-height/2,width,height);
    painter.drawImage(picRect,mImage);

    QWidget::paintEvent(event);
}

void ImageTransformLabel::wheelEvent(QWheelEvent *event)
{
    int value = event->delta();
    if(value > 0)//放大
    {
        slot_ActionZoomInImage();//内部已调用了update
    }
    else//缩小
    {
        slot_ActionZoomOutImage();//内部已调用了update
    }

}

void ImageTransformLabel::mousePressEvent(QMouseEvent *event)
{
    mOldPos = event->pos();
    mPressedFlag = true;
    this->setCursor(Qt::ClosedHandCursor);//设置鼠标样式
}

void ImageTransformLabel::mouseMoveEvent(QMouseEvent *event)
{
    if(!mPressedFlag)
        return QWidget::mouseMoveEvent(event);

    QPoint pos = event->pos();
    int offsetX = pos.x() - mOldPos.x();
    int offsetY = pos.y() - mOldPos.y();

    mOffsetX += offsetX;
    mOffsetY += offsetY;

    mOldPos = pos;

    update();
}

void ImageTransformLabel::mouseReleaseEvent(QMouseEvent *event)
{
    mPressedFlag = false;
    this->setCursor(Qt::ArrowCursor);//设置鼠标样式
}

void ImageTransformLabel::slot_ActionLoadImage()
{
    QString imagePath = QFileDialog::getOpenFileName(this,"选择图像","./",tr("Images (*.png *.jpg *.jpeg *.bmp)"));
    if(imagePath.isEmpty())
        return;

    setImage(imagePath);

}

void ImageTransformLabel::slot_ActionZoomInImage()
{
    mZoomValue += 0.05;
    update();
}

void ImageTransformLabel::slot_ActionZoomOutImage()
{
    mZoomValue -= 0.05;
    if(mZoomValue < 0)
    {
        mZoomValue = 0.05;
    }
    update();
}

void ImageTransformLabel::slot_ActionPresetImage()
{
    mZoomValue = 1.0;
    mOffsetX = 0;
    mOffsetY = 0;
    update();
}

void ImageTransformLabel::slot_ActionClearImage()
{
    this->clear();
    mImage = QImage();
    mImagePath = "";
}

备注:调用时对QLable类对象进行提升即可

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值