QT 通过鼠标事件实现图片的拖动和缩放

通过鼠标拖动来移动图片,并使用鼠标滚轮来缩放图片。

1、实现步骤:

1、移动图片:

bash 复制代码
使用QPoint记录图片的偏移量,当鼠标拖动时更新这个偏移量,在paintEvent()中根据偏移量绘制图片。

2、缩放图片:

bash 复制代码
使用滚轮事件调整图片的缩放倍数,在paintEvent()中重新绘制图片时应用缩放。

2、完整的解决方案:

1、ImageWidget.h

cpp 复制代码
#ifndef IMAGEWIDGET_H
#define IMAGEWIDGET_H

#include <QApplication>
#include <QWidget>
#include <QLabel>
#include <QPixmap>
#include <QMouseEvent>
#include <QWheelEvent>
#include <QPainter>
#include <QVBoxLayout>
#include <QDebug>


class ImageWidget : public QWidget
{
    Q_OBJECT

public:
    ImageWidget(QWidget *parent = nullptr);
protected:
    // 重写paintEvent,用于绘制图片
    void paintEvent(QPaintEvent *event) override;

    // 鼠标按下事件
    void mousePressEvent(QMouseEvent *event) override;

    // 鼠标移动事件
    void mouseMoveEvent(QMouseEvent *event) override;
    // 鼠标释放事件
    void mouseReleaseEvent(QMouseEvent *event) override;
    // 滚轮事件,用于缩放图片
    void wheelEvent(QWheelEvent *event) override;

private:
    QPixmap pixmap;            // 图片
    QPoint lastMousePosition;  // 上一次鼠标点击的位置
    QPoint offset;             // 图片的偏移量
    bool isDragging;           // 标记是否正在拖动图片
    double scaleFactor;        // 缩放倍数
};


#endif // IMAGEWIDGET_H

2、ImageWidget.cpp

bash 复制代码
#include "imagewidget.h"


ImageWidget::ImageWidget(QWidget *parent) : QWidget(parent)
{
    // 加载图片
    pixmap = QPixmap(R"(C:\Users\LiangQL\Desktop\开发者基础认证.jpg)");  // 替换成你的图片路径
    scaleFactor = 1.0;        // 初始缩放因子为1
    isDragging = false;       // 初始状态下不拖动
    offset = QPoint(0, 0);    // 图片的初始偏移为(0, 0)
}
// 重写paintEvent,用于绘制图片
void ImageWidget::paintEvent(QPaintEvent *event)
{
    // 创建一个QPainter,用于绘制图片
    QPainter painter(this);
    painter.setRenderHint(QPainter::SmoothPixmapTransform);  // 开启平滑缩放

    // 计算缩放后的图片尺寸
    QSize scaledSize = pixmap.size() * scaleFactor;

    // 绘制图片,应用缩放和偏移量
    painter.drawPixmap(offset, pixmap.scaled(scaledSize, Qt::KeepAspectRatio, Qt::SmoothTransformation));
}


// 鼠标按下事件
void ImageWidget::mousePressEvent(QMouseEvent *event)
{
    if (event->button() == Qt::LeftButton) {
        isDragging = true;                // 开始拖动
        lastMousePosition = event->pos(); // 记录鼠标点击位置
    }
}

// 鼠标移动事件
void ImageWidget::mouseMoveEvent(QMouseEvent *event)
{
    if (isDragging) {
        // 计算鼠标的移动距离
        QPoint delta = event->pos() - lastMousePosition;

        // 更新图片的偏移量
        offset += delta;

        // 更新鼠标位置
        lastMousePosition = event->pos();

        // 触发重新绘制
        update();
    }
}

// 鼠标释放事件
void ImageWidget::mouseReleaseEvent(QMouseEvent *event)
{
    if (event->button() == Qt::LeftButton) {
        isDragging = false;  // 结束拖动
    }
}

// 滚轮事件,用于缩放图片
void ImageWidget::wheelEvent(QWheelEvent *event)
{
    // 滚轮向上放大,向下缩小
    if (event->angleDelta().y() > 0) {
        scaleFactor *= 1.1;  // 放大
    } else {
        scaleFactor /= 1.1;  // 缩小
    }

    // 限制缩放范围
    scaleFactor = qBound(0.1, scaleFactor, 10.0);

    // 更新显示
    update();
}

3、 调用main.cpp

bash 复制代码
#include <QApplication>
#include <QWidget>
#include <QLabel>
#include <QPixmap>
#include <QMouseEvent>
#include <QWheelEvent>
#include <QPainter>
#include <QVBoxLayout>
#include <QDebug>
#include "ImageWidget.h"

int main(int argc, char *argv[])
{
    QApplication app(argc, argv);

    // 创建主窗口
    QWidget window;
    window.setFixedSize(800, 600);  // 设置窗口大小

    // 创建图片控件(自定义的类)
    ImageWidget *imageWidget = new ImageWidget;

    // 使用布局管理器将图片控件放置在窗口中
    QVBoxLayout *layout = new QVBoxLayout(&window);
    layout->addWidget(imageWidget);

    // 显示主窗口
    window.show();

    return app.exec();
}

3、详细解释:

1. paintEvent() 重绘图片:

bash 复制代码
paintEvent()方法用于自定义控件的绘制。在这个方法中,我们使用QPainter对象来绘制图片。
pixmap.scaled():使用pixmap.scaled()方法来缩放图片,根据scaleFactor缩放图片并保持其宽高比。
painter.drawPixmap(offset, ...):在指定的偏移量位置绘制图片,offset表示图片在控件中的偏移。

2. 图片移动:

bash 复制代码
mousePressEvent():当鼠标左键按下时,记录鼠标点击位置,并开始拖动图片(isDragging = true)。
mouseMoveEvent():如果正在拖动图片(isDragging为true),则计算鼠标的移动距离,将图片的offset偏移量更新为原来的偏移量加上鼠标移动的距离。
mouseReleaseEvent():当鼠标左键释放时,结束拖动(isDragging = false)。

3. 图片缩放:

bash 复制代码
wheelEvent():处理鼠标滚轮事件。滚轮向上时,放大图片;滚轮向下时,缩小图片。
scaleFactor *= 1.1:每次滚动时,缩放倍数按10%变化。
qBound(0.1, scaleFactor, 10.0):限制缩放比例在0.1到10倍之间,防止图片缩得太小或放得太大。

4. 平滑缩放:

bash 复制代码
Qt::SmoothTransformation:使用平滑缩放算法,防止图片缩放后出现锯齿。

5. 偏移量的作用:

bash 复制代码
offset用于记录图片相对于窗口的偏移量,当鼠标拖动时,更新这个偏移量。
在paintEvent()中,每次绘制时都使用这个偏移量来控制图片的位置。
运行效果:
拖动图片:按住鼠标左键并移动,可以看到图片在窗口内部移动。
缩放图片:滚动鼠标滚轮,图片会放大或缩小,同时保持宽高比。
相关推荐
火星MARK1 分钟前
RAID详解
数据库·oracle
JAVA学习通2 分钟前
Spring AI与DeepSeek实战:打造企业级智能体
数据库
安审若无1 小时前
Oracle 打补丁指南
数据库·oracle
樱花的浪漫1 小时前
Cuda reduce算子实现与优化
数据库·人工智能·深度学习·神经网络·机器学习·自然语言处理
啊森要自信1 小时前
【MySQL 数据库】MySQL用户管理
android·c语言·开发语言·数据库·mysql
kkkkk0211061 小时前
Redis八股
数据库·redis·缓存
Liu1bo2 小时前
【MySQL】表的约束
linux·数据库·mysql
胖胖的战士2 小时前
Mysql 数据库迁移
数据库·mysql
czhc11400756633 小时前
LINUX1012 mysql GLIBC安装
数据库·mysql
MC皮蛋侠客3 小时前
Ubuntu禁用系统手势,阻止应用程序异常最小化
linux·运维·qt·ubuntu