Qt QGraphics类应用——图片移动+选点缩放+控制移动区域

记录:本博客用来记录学习,引用博客https://blog.csdn.net/weixin_43935474/article/details/89327314

功能:

  1、图片可以使用鼠标进行拖动

  2、图片可以在鼠标位置放大缩小

  3、图片移动的区域固定

需求:

  图片一张

成品:

 

代码:

ImageWidget.h

 1 #ifndef IMAGEWIDGET_H
 2 #define IMAGEWIDGET_H
 3 
 4 #include <QWidget>
 5 #include <QtGui>
 6 #include <QPixmap>
 7 #include <QPainter>
 8 #include <QRectF>
 9 #include <QMouseEvent>
10 #include <QPointF>
11 #include <QDragEnterEvent>
12 #include <QGraphicsSceneWheelEvent>
13 #include <QGraphicsItem>
14 
15 enum Enum_ZoomState{
16     NO_STATE,
17     RESET,
18     ZOOM_IN,
19     ZOOM_OUT
20 };
21 // class ImageWidget :public QObject, QGraphicsItem
22 class ImageWidget :public QGraphicsItem
23 {
24     //Q_OBJECT
25 public:
26     ImageWidget(QPixmap *pixmap);
27     QRectF  boundingRect() const;
28     void    paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget);
29     void    wheelEvent(QGraphicsSceneWheelEvent *event);
30     void    ResetItemPos();
31     //QVariant itemChange(GraphicsItemChange change, const QVariant &value);
32     void    mousePressEvent(QGraphicsSceneMouseEvent *event);
33     void    mouseMoveEvent(QGraphicsSceneMouseEvent *event);
34     void    mouseReleaseEvent(QGraphicsSceneMouseEvent *event);
35     qreal   getScaleValue() const;
36     void    setQGraphicsViewWH(int nwidth,int nheight);
37 private:
38     qreal       m_scaleValue;
39     qreal       m_scaleDafault;
40     QPixmap     m_pix;
41     int         m_zoomState;
42     bool        m_isMove;
43     QPointF     m_startPos;
44 };
45 #endif // IMAGEWIDGET_H

ImageWidget.cpp

#include "imagewidget.h"
#include <QDebug>
#include <QGraphicsSceneMouseEvent>
#include <QPointF>
#include <QGraphicsSceneDragDropEvent>
#include <QDrag>
#include <math.h>

ImageWidget::ImageWidget(QPixmap *pixmap)
{
    m_pix = *pixmap;
    //If enabled is true, this item will accept hover events; otherwise, it will ignore them. By default, items do not accept hover events.
    setAcceptDrops(true);//true接收悬停事件
    m_scaleValue = 0;
    m_scaleDafault = 0;
    m_isMove = false;
}

QRectF ImageWidget::boundingRect() const
{
    return QRectF(-m_pix.width() / 2, -m_pix.height() / 2,
                  m_pix.width(), m_pix.height());
}

void ImageWidget::paint(QPainter *painter, const QStyleOptionGraphicsItem *,
                    QWidget *)
{
    painter->drawPixmap(-m_pix.width() / 2, -m_pix.height() / 2, m_pix);
}

void ImageWidget::mousePressEvent(QGraphicsSceneMouseEvent *event)
{
    if(event->button()== Qt::LeftButton)
    {
        m_startPos = event->pos();//鼠标左击时,获取当前鼠标在图片中的坐标,
        m_isMove = true;//标记鼠标左键被按下
    }
    else if(event->button() == Qt::RightButton)
    {
        ResetItemPos();//右击鼠标重置大小
    }

}

void ImageWidget::mouseMoveEvent(QGraphicsSceneMouseEvent *event)
{
    qreal x = m_pix.width()/4;
    qreal y = m_pix.height()/4;

        if(pos().x() <= (-x))
        {
            setPos(qreal(-x+1),pos().y());
            m_isMove = false;
        }
        if(pos().x() >= x)
        {
            setPos(QPointF(x-1,pos().y()));
            m_isMove = false;
        }
        if(pos().y() <= (-y))
        {
            setPos(pos().x(),qreal(-y+1));
            m_isMove = false;
        }
        if(pos().y() >= y)
        {
            setPos(pos().x(),qreal(y-1));
            m_isMove = false;
        }
    if(m_isMove)
    {
        QPointF point = (event->pos() - m_startPos)*m_scaleValue;
        moveBy(point.x(), point.y());
    }
    qDebug()<<"m_pix.width()/3="<<m_pix.width()/3;
    qDebug()<<"m_pix.height()/3="<<m_pix.height()/3;
    qDebug()<<"event->pos="<<event->pos();
    qDebug()<<"pos="<<pos();
}

void ImageWidget::mouseReleaseEvent(QGraphicsSceneMouseEvent *)
{
    m_isMove = false;//标记鼠标左键已经抬起
}


void ImageWidget::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);//使图片缩放的效果看起来像是以鼠标所在点为中心进行缩放的
        }
    }
}

void ImageWidget::setQGraphicsViewWH(int nwidth, int nheight)//将主界面的控件QGraphicsView的width和height传进本类中,并根据图像的长宽和控件的长宽的比例来使图片缩放到适合控件的大小
{
    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 ImageWidget::ResetItemPos()//重置图片位置
{
    m_scaleValue = m_scaleDafault;//缩放比例回到一开始的自适应比例
    setScale(m_scaleDafault);//缩放到一开始的自适应大小
    setPos(0,0);
}

qreal ImageWidget::getScaleValue() const
{
    return m_scaleValue;
}

  

mainwindow.h

#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QMainWindow>
#include "imagewidget.h"

namespace Ui {
class MainWindow;
}

class MainWindow : public QMainWindow
{
    Q_OBJECT
    
public:
    explicit MainWindow(QWidget *parent = 0);
    ~MainWindow();
    void            recvShowPicSignal(QImage image);//接收并显示图片的函数
private:
    Ui::MainWindow *ui;
    ImageWidget    *m_Image;
};

#endif // MAINWINDOW_H

mainwindow.cpp

#include "mainwindow.h"
#include "ui_mainwindow.h"
#include <QGraphicsItem>

MainWindow::MainWindow(QWidget *parent) :
    QMainWindow(parent),
    ui(new Ui::MainWindow)
{
    ui->setupUi(this);
    recvShowPicSignal(QImage(QString::fromUtf8(":/image/但丁真魔人.jpg")));
}

MainWindow::~MainWindow()
{
    delete ui;
}

void MainWindow::recvShowPicSignal(QImage image)
{

    QPixmap ConvertPixmap=QPixmap::fromImage(image);
    int width = ui->ImageGraphic->width();//获取界面控件Graphics View的宽度
    int height = ui->ImageGraphic->height();//获取界面控件Graphics View的高度

    qDebug()<<"width="<<width<<" heigth="<<height<<" size="<< ui->ImageGraphic->size();
    //要用QGraphicsView就必须要有QGraphicsScene搭配着用
    QGraphicsScene  *qgraphicsScene = new QGraphicsScene;
    qgraphicsScene->setSceneRect(QRectF(-width/2,-height/2,width,height));

    qDebug()<<"width="<<width<<" height="<<height;
    //实例化类ImageWidget的对象m_Image,该类继承自QGraphicsItem,是自己写的类
    m_Image = new ImageWidget(&ConvertPixmap);
    //将界面控件Graphics View的width和height传进类m_Image中
    m_Image->setQGraphicsViewWH(width,height);

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

    //使视窗的大小固定在原始大小,不会随图片的放大而放大(默认状态下图片放大的时候视窗两边会自动出现滚动条,
    //并且视窗内的视野会变大),防止图片放大后重新缩小的时候视窗太大而不方便观察图片
    ui->ImageGraphic->setSceneRect(QRectF(-(width/2),-(height/2),width,height));//使视窗的大小固定在原始大小,不会随图片的放大而放大(默认状态下图片放大的时候视窗两边会自动出现滚动条,并且视窗内的视野会变大),防止图片放大后重新缩小的时候视窗太大而不方便观察图片
    //ui->ImageGraphic->setMinimumSize(width,height);
    ui->ImageGraphic->setScene(qgraphicsScene);
    ui->ImageGraphic->setFocus();//将界面的焦点设置到当前Graphics View控件

}

  

posted @ 2019-12-11 17:06  硕果磊磊  阅读(5898)  评论(2编辑  收藏  举报