QGraphicsView实现拖拽缩放

2024-04-09 20:12

本文主要是介绍QGraphicsView实现拖拽缩放,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

QGraphicsView实现拖拽缩放_qt qgraphicview 视图缩放-CSDN博客

首先创建视图框架,分别是QGraphicsView、QGraphicsScene和QGraphicsItem。
其中QGraphicsItem需要继承重写,重写的派生类中必须需要实现两个函数,paint(item的绘制函数)和boundingRect(item的大小位置函数),因为这两个是纯虚函数。

然后在view视图中添加Scene。设置view的
setDragMode打开拖拽功能。设置view的setTransform实现缩放功能。

示例代码如下,该示例支持按钮缩放,滚轮缩放和鼠标拖拽。

#ifndef WIDGET_H
#define WIDGET_H#include <QWidget>
#include "mgraaphicsview.h"class Widget : public QWidget
{Q_OBJECTpublic:explicit Widget(QWidget *parent = nullptr);~Widget();private:MGraaphicsView* m_view = nullptr;
};#endif // WIDGET_H
#include "widget.h"#include <QPushButton>
#include <QGridLayout>
#include <QDebug>Widget::Widget(QWidget *parent) :QWidget(parent)
{QPushButton* bbtn = new QPushButton("放大",this);QPushButton* sbtn = new QPushButton("缩小",this);QPushButton* obtn = new QPushButton("还原",this);QGraphicsScene* scene = new QGraphicsScene();m_view = new MGraaphicsView(scene);QGridLayout* lay = new QGridLayout(this);lay->addWidget(m_view,0,0,10,10);lay->addWidget(bbtn,10,0,1,1);lay->addWidget(sbtn,10,1,1,1);lay->addWidget(obtn,10,2,1,1);lay->setMargin(0);this->setLayout(lay);connect(bbtn,&QPushButton::clicked,this,[=](){m_view->zoomOnce(1);});connect(sbtn,&QPushButton::clicked,this,[=](){m_view->zoomOnce(-1);});connect(obtn,&QPushButton::clicked,this,[=](){});}Widget::~Widget()
{
}

view视图框架类,相当于显示窗口,给它添加QGraphicsScene,使其拥有界面,且可以添加多个QGraphicsScene。
setHorizontalScrollBarPolicy(Qt::ScrollBarPolicy::ScrollBarAlwaysOff);和
setVerticalScrollBarPolicy(Qt::ScrollBarPolicy::ScrollBarAlwaysOff);隐藏滚动条。

#ifndef MGRAAPHICSVIEW_H
#define MGRAAPHICSVIEW_H#include <QGraphicsView>
#include <QWheelEvent>
#include "mgraphicsitem.h"class MGraaphicsView : public QGraphicsView
{
public:MGraaphicsView(QGraphicsScene *scene, QWidget *parent = nullptr);void wheelEvent(QWheelEvent *event);void zoomOnce(int val);int m_minZoom;int m_maxZoom;int m_zoom = 0;double m_zoomnum = 1;MGraphicsItem* _item = nullptr;
};#endif // MGRAAPHICSVIEW_H
#include "mgraaphicsview.h"
#include <qmath.h>
#include <QDebug>MGraaphicsView::MGraaphicsView(QGraphicsScene *scene, QWidget *parent):QGraphicsView(scene,parent)
{setHorizontalScrollBarPolicy(Qt::ScrollBarPolicy::ScrollBarAlwaysOff);setVerticalScrollBarPolicy(Qt::ScrollBarPolicy::ScrollBarAlwaysOff);setDragMode(QGraphicsView::ScrollHandDrag);setTransformationAnchor(QGraphicsView::AnchorUnderMouse);_item = new MGraphicsItem();_item->setboundingRect(0,0,1000,1000);scene->addItem(_item);
}void MGraaphicsView::wheelEvent(QWheelEvent *event)
{event->accept();zoomOnce(event->delta());
}void MGraaphicsView::zoomOnce(int val)
{if (val > 0){m_zoom++;auto scaleValue = qPow(2, m_zoom);setTransform(QTransform::fromScale(scaleValue, scaleValue));}else{m_zoom--;auto scaleValue = qPow(2, m_zoom);setTransform(QTransform::fromScale(scaleValue, scaleValue));}
}

QGraphicsItem类,它是界面上的控件,相对于Qt的标准控件,它更省内存。且在中QGraphicsScene中也可以添加QT标准控件。通过addWidget即可。如下

		QWidget *w = new QWidget ()QGraphicsProxyWidget* pwidget = m_mapView->scene()->addWidget(w);pwidget->setFlag(QGraphicsItem::ItemIgnoresTransformations, true); //禁止随界面缩放而缩放pwidget->setFlag(QGraphicsItem::ItemSendsGeometryChanges, true); //禁止随界面大小变化而变化

继承QGraphicsItem实现的自定义Item,由于QGraphicsItem它不继承QObject,所有它不具备信号槽机制,如果想要实现信号槽,可以多重继承,既继承QObject,又继承QGraphicsItem,就像QGraphicsObject它一样。

#ifndef MGRAPHICSITEM_H
#define MGRAPHICSITEM_H#include <QGraphicsItem>
#include <QPainter>
#include <QSvgRenderer>class MGraphicsItem : public QGraphicsItem
{
public:MGraphicsItem();virtual void paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget = nullptr);virtual QRectF boundingRect() const ;void setboundingRect(int x, int y, int w, int h);int m_x = 0;int m_y = 0;int m_w = 0;int m_h = 0;QSvgRenderer* _renderer = nullptr;void reLoad(QString name);
};#endif // MGRAPHICSITEM_H
#include "mgraphicsitem.h"MGraphicsItem::MGraphicsItem()
{_renderer = new QSvgRenderer;_renderer->load(QString("test.svg"));
}void MGraphicsItem::reLoad(QString name)
{_renderer->load(name);
}void MGraphicsItem::setboundingRect(int x,int y,int w,int h)
{m_x = x;m_y = y;m_w = w;m_h = h;
}void MGraphicsItem::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget)
{if (!_renderer->isValid())return;_renderer->render(painter, boundingRect());
}QRectF MGraphicsItem::boundingRect() const
{return QRectF(m_x,m_y,m_w,m_h);
}

QSvgRenderer它是用来显示svg矢量图的一个代理类。

这篇关于QGraphicsView实现拖拽缩放的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/889097

相关文章

C++中零拷贝的多种实现方式

《C++中零拷贝的多种实现方式》本文主要介绍了C++中零拷贝的实现示例,旨在在减少数据在内存中的不必要复制,从而提高程序性能、降低内存使用并减少CPU消耗,零拷贝技术通过多种方式实现,下面就来了解一下... 目录一、C++中零拷贝技术的核心概念二、std::string_view 简介三、std::stri

C++高效内存池实现减少动态分配开销的解决方案

《C++高效内存池实现减少动态分配开销的解决方案》C++动态内存分配存在系统调用开销、碎片化和锁竞争等性能问题,内存池通过预分配、分块管理和缓存复用解决这些问题,下面就来了解一下... 目录一、C++内存分配的性能挑战二、内存池技术的核心原理三、主流内存池实现:TCMalloc与Jemalloc1. TCM

OpenCV实现实时颜色检测的示例

《OpenCV实现实时颜色检测的示例》本文主要介绍了OpenCV实现实时颜色检测的示例,通过HSV色彩空间转换和色调范围判断实现红黄绿蓝颜色检测,包含视频捕捉、区域标记、颜色分析等功能,具有一定的参考... 目录一、引言二、系统概述三、代码解析1. 导入库2. 颜色识别函数3. 主程序循环四、HSV色彩空间

Python实现精准提取 PDF中的文本,表格与图片

《Python实现精准提取PDF中的文本,表格与图片》在实际的系统开发中,处理PDF文件不仅限于读取整页文本,还有提取文档中的表格数据,图片或特定区域的内容,下面我们来看看如何使用Python实... 目录安装 python 库提取 PDF 文本内容:获取整页文本与指定区域内容获取页面上的所有文本内容获取

基于Python实现一个Windows Tree命令工具

《基于Python实现一个WindowsTree命令工具》今天想要在Windows平台的CMD命令终端窗口中使用像Linux下的tree命令,打印一下目录结构层级树,然而还真有tree命令,但是发现... 目录引言实现代码使用说明可用选项示例用法功能特点添加到环境变量方法一:创建批处理文件并添加到PATH1

Java使用HttpClient实现图片下载与本地保存功能

《Java使用HttpClient实现图片下载与本地保存功能》在当今数字化时代,网络资源的获取与处理已成为软件开发中的常见需求,其中,图片作为网络上最常见的资源之一,其下载与保存功能在许多应用场景中都... 目录引言一、Apache HttpClient简介二、技术栈与环境准备三、实现图片下载与保存功能1.

canal实现mysql数据同步的详细过程

《canal实现mysql数据同步的详细过程》:本文主要介绍canal实现mysql数据同步的详细过程,本文通过实例图文相结合给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的... 目录1、canal下载2、mysql同步用户创建和授权3、canal admin安装和启动4、canal

Nexus安装和启动的实现教程

《Nexus安装和启动的实现教程》:本文主要介绍Nexus安装和启动的实现教程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、Nexus下载二、Nexus安装和启动三、关闭Nexus总结一、Nexus下载官方下载链接:DownloadWindows系统根

SpringBoot集成LiteFlow实现轻量级工作流引擎的详细过程

《SpringBoot集成LiteFlow实现轻量级工作流引擎的详细过程》LiteFlow是一款专注于逻辑驱动流程编排的轻量级框架,它以组件化方式快速构建和执行业务流程,有效解耦复杂业务逻辑,下面给大... 目录一、基础概念1.1 组件(Component)1.2 规则(Rule)1.3 上下文(Conte

MySQL 横向衍生表(Lateral Derived Tables)的实现

《MySQL横向衍生表(LateralDerivedTables)的实现》横向衍生表适用于在需要通过子查询获取中间结果集的场景,相对于普通衍生表,横向衍生表可以引用在其之前出现过的表名,本文就来... 目录一、横向衍生表用法示例1.1 用法示例1.2 使用建议前面我们介绍过mysql中的衍生表(From子句