Qt利用QGraphicsView类实现图片放大缩小平移显示

0 1866
索鸟 2021-04-23
需要:0索币

写一个类来继承QGraphicsItem,来控制图片的平移放大缩小等操作

#pragma once

#include <QWidget>

#include <QtGui>

#include <QPixmap>

#include <QPainter>

#include <QRectF>

#include <QMouseEvent>

#include <QPointF>

#include <QDragEnterEvent>

#include <QGraphicsSceneWheelEvent>

#include <QGraphicsItem>

enum Enum_ZoomState {

    NO_STATE,

    RESET,

    ZOOM_IN,

    ZOOM_OUT

};

class QImageWidget :

    public QGraphicsItem

{

public:

    QImageWidget(QPixmap* pixmap);

    QRectF  boundingRect() const;

    void    paint(QPainter* painter, const QStyleOptionGraphicsItem* option, QWidget* widget);

    void    wheelEvent(QGraphicsSceneWheelEvent* event);

    void    ResetItemPos();

    void    mousePressEvent(QGraphicsSceneMouseEvent* event);

    void    mouseMoveEvent(QGraphicsSceneMouseEvent* event);

    void    mouseReleaseEvent(QGraphicsSceneMouseEvent* event);

    qreal   getScaleValue() const;

    void    setQGraphicsViewWH(int nwidth, int nheight);

private:

    qreal       m_scaleValue;

    qreal       m_scaleDafault;

    QPixmap     m_pix;

    int         m_zoomState;

    bool        m_isMove;

    QPointF     m_startPos;

};

#include "QImageWidget.h"

#include <QDebug>

#include <QGraphicsSceneMouseEvent>

#include <QPointF>

#include <QGraphicsSceneDragDropEvent>

#include <QDrag>

#include <math.h>

QImageWidget::QImageWidget(QPixmap* pixmap)

{

    m_pix = *pixmap;

    setAcceptDrops(true);//If enabled is true, this item will accept hover events; otherwise, it will ignore them. By default, items do not accept hover events.

    m_scaleValue = 0;

    m_scaleDafault = 0;

    m_isMove = false;

}

QRectF QImageWidget::boundingRect() const

{

    return QRectF(-m_pix.width() / 2, -m_pix.height() / 2,

        m_pix.width(), m_pix.height());

}

void QImageWidget::paint(QPainter* painter, const QStyleOptionGraphicsItem*,

    QWidget*)

{

    painter->drawPixmap(-m_pix.width() / 2, -m_pix.height() / 2, m_pix);

}

void QImageWidget::mousePressEvent(QGraphicsSceneMouseEvent* event)

{

    if (event->button() == Qt::LeftButton)

    {

        m_startPos = event->pos();//鼠标左击时,获取当前鼠标在图片中的坐标,

        m_isMove = true;//标记鼠标左键被按下

    }

    else if (event->button() == Qt::RightButton)

    {

        ResetItemPos();//右击鼠标重置大小

    }

}

void QImageWidget::mouseMoveEvent(QGraphicsSceneMouseEvent* event)

{

    if (m_isMove)

    {

        QPointF point = (event->pos() - m_startPos) * m_scaleValue;

        moveBy(point.x(), point.y());

    }

}

void QImageWidget::mouseReleaseEvent(QGraphicsSceneMouseEvent*)

{

    m_isMove = false;//标记鼠标左键已经抬起

}

void QImageWidget::wheelEvent(QGraphicsSceneWheelEvent* event)//鼠标滚轮事件

{

    if ((event->delta() > 0) && (m_scaleValue >= 50))//最大放大到原始图像的50倍

    {

        return;

    }

    else if ((event->delta() < 0) && (m_scaleValue <= m_scaleDafault))//图像缩小到自适应大小之后就不继续缩小

    {

        ResetItemPos();//重置图片大小和位置,使之自适应控件窗口大小

    }

    else

    {

        qreal qrealOriginScale = m_scaleValue;

        if (event->delta() > 0)//鼠标滚轮向前滚动

        {

            m_scaleValue *= 1.1;//每次放大10%

        }

        else

        {

            m_scaleValue *= 0.9;//每次缩小10%

        }

        setScale(m_scaleValue);

        if (event->delta() > 0)

        {

            moveBy(-event->pos().x() * qrealOriginScale * 0.1, -event->pos().y() * qrealOriginScale * 0.1);//使图片缩放的效果看起来像是以鼠标所在点为中心进行缩放的

        }

        else

        {

            moveBy(event->pos().x() * qrealOriginScale * 0.1, event->pos().y() * qrealOriginScale * 0.1);//使图片缩放的效果看起来像是以鼠标所在点为中心进行缩放的

        }

    }

}

//将主界面的控件QGraphicsView的width和height传进本类中,并根据图像的长宽和控件的长宽的比例来使图片缩放到适合控件的大小

void QImageWidget::setQGraphicsViewWH(int nwidth, int nheight)

{

    int nImgWidth = m_pix.width();

    int nImgHeight = m_pix.height();

    qreal temp1 = nwidth * 1.0 / nImgWidth;

    qreal temp2 = nheight * 1.0 / nImgHeight;

    if (temp1 > temp2)

    {

        m_scaleDafault = temp2;

    }

    else

    {

        m_scaleDafault = temp1;

    }

    setScale(m_scaleDafault);

    m_scaleValue = m_scaleDafault;

}

//重置图片位置

void QImageWidget::ResetItemPos()

{

    m_scaleValue = m_scaleDafault;//缩放比例回到一开始的自适应比例

    setScale(m_scaleDafault);//缩放到一开始的自适应大小

    setPos(0, 0);

}

qreal QImageWidget::getScaleValue() const

{

    return m_scaleValue;

}

再主窗口进行打开图片显示

打开图片操作如下

//打开图片文件

void OpenCVTestSf::OpenImageFile()

{

    QString fileName = QFileDialog::getOpenFileName(this, "open Image", "", "Image File(*.bmp *.jpg *.jpeg *.png)");

    QTextCodec* code = QTextCodec::codecForName("gb18030");

    std::string name = code->fromUnicode(fileName).data();

    m_srcImage = imread(name);

    if (m_srcImage.data)

    {

        cvtColor(m_srcImage, m_srcImage, COLOR_BGR2RGB);//BGR转化为RGB

                QImage::Format format = QImage::Format_RGB888;

        switch (m_srcImage.type())

        {

        case CV_8UC1:

            format = QImage::Format_Indexed8;

            break;

        case CV_8UC3:

            format = QImage::Format_RGB888;

            break;

        case CV_8UC4:

            format = QImage::Format_ARGB32;

            break;

        }

        QImage img = QImage((const uchar*)m_srcImage.data, m_srcImage.cols, m_srcImage.rows,

            m_srcImage.cols * m_srcImage.channels(), format);

        recvShowPicSignal(img);


    }

}

显示图片如下

//显示图片

void OpenCVTestSf::recvShowPicSignal(QImage image)

{

    QPixmap ConvertPixmap = QPixmap::fromImage(image);//The QPixmap class is an off-screen image representation that can be used as a paint device

    QGraphicsScene* qgraphicsScene = new QGraphicsScene;//要用QGraphicsView就必须要有QGraphicsScene搭配着用

    m_pImage = new QImageWidget(&ConvertPixmap);//实例化类ImageWidget的对象m_Image,该类继承自QGraphicsItem,是自己写的类

    int nwith = ui.graphicsViewImgShow->width();//获取界面控件Graphics View的宽度

    int nheight = ui.graphicsViewImgShow->height();//获取界面控件Graphics View的高度

    m_pImage->setQGraphicsViewWH(nwith, nheight);//将界面控件Graphics View的width和height传进类m_Image中

    qgraphicsScene->addItem(m_pImage);//将QGraphicsItem类对象放进QGraphicsScene中

    ui.graphicsViewImgShow->setSceneRect(QRectF(-(nwith / 2), -(nheight / 2), nwith, nheight));//使视窗的大小固定在原始大小,不会随图片的放大而放大(默认状态下图片放大的时候视窗两边会自动出现滚动条,并且视窗内的视野会变大),防止图片放大后重新缩小的时候视窗太大而不方便观察图片

    ui.graphicsViewImgShow->setScene(qgraphicsScene);//Sets the current scene to scene. If scene is already being viewed, this function does nothing.

    ui.graphicsViewImgShow->setFocus();//将界面的焦点设置到当前Graphics View控件

}

原文 https://mp.weixin.qq.com/s/GLwR1cr-kmBr7eNJ2abFjg

回帖
  • 消灭零回复
相关主题
2020年最新最新Kubernetes视频教程(K8s)教程 2
程序员转型之制作网课变现,月入过万告别996 1
索鸟快传2.0发布啦 1
两个不同网络的电脑怎么实现文件的互相访问呢? 1
网盘多账号登录软件 1
Java实战闲云旅游项目基于vue+element-ui 1
单点登录技术解决方案基于OAuth2.0的网关鉴权RSA算法生成令牌 1
QT5获取剪贴板上文本信息QT设置剪贴板内容 1
springboot2实战在线购物系统电商系统 1
python web实战之爱家租房项目 1
windows COM实用入门教程 1
C++游戏开发之C++实现的水果忍者游戏 1
计算机视觉库opencv教程 1
node.js实战图书管理系统express框架实现 1
C++实战教程之远程桌面远程控制实战 1
相关主题
PHP7报A non well formed numeric value encountered 0
Linux系统下关闭mongodb的几种命令分享 0
mongodb删除数据、删除集合、删除数据库的命令 0
Git&Github极速入门与攻坚实战课程 0
python爬虫教程使用Django和scrapy实现 0
libnetsnmpmibs.so.31: cannot open shared object file 0
数据结构和算法视频教程 0
redis的hash结构怎么删除数据呢? 0
C++和LUA解析器的数据交互实战视频 0
mongodb errmsg" : "too many users are authenticated 0
C++基础入门视频教程 0
用30个小时精通C++视频教程可能吗? 0
C++分布式多线程游戏服务器开发视频教程socket tcp boost库 0
C++培训教程就业班教程 0
layui的util工具格式时间戳为字符串 0
C++实战教程之远程桌面远程控制实战 1
网络安全培训视频教程 0
LINUX_C++软件工程师视频教程高级项目实战 0
C++高级数据结构与算法视频教程 0
跨域问题很头疼?通过配置nginx轻松解决ajax跨域问题 0