QGraphicsView 实现图片裁剪功能

2024-04-01 01:08

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

本教程实现了图片显示、滚轮放大缩小图片、鼠标中键移动图片、画框裁剪图片功能。

界面程序

from PyQt5.QtCore import QSize, Qt, QRect
from PyQt5.QtGui import QColor
from PyQt5.QtWidgets import QApplication, QDialog, QGridLayout, QPushButton, QSpacerItem, QSizePolicytry:from .graphics import GraphicsView, GraphicsPixmapItem
except:from graphics import GraphicsView, GraphicsPixmapItemclass Form(QDialog):def __init__(self):super(Form, self).__init__()self.resize(1024, 768)self.picture = r'test.png'self.init_ui()# 视图背景颜色self.graphicsView.setBackgroundBrush(QColor(28, 31, 34))self.graphicsView.save_signal.connect(self.pushButton_save.setEnabled)self.pushButton_cut.clicked.connect(self.pushButton_cut_clicked)self.pushButton_save.clicked.connect(self.pushButton_save_clicked)# image_item = GraphicsPolygonItem()# image_item.setFlag(QGraphicsItem.ItemIsMovable)# self.scene.addItem(image_item)def init_ui(self):self.gridLayout = QGridLayout(self)self.pushButton_cut = QPushButton('cut', self)self.pushButton_cut.setCheckable(True)self.pushButton_cut.setMaximumSize(QSize(100, 16777215))self.gridLayout.addWidget(self.pushButton_cut, 0, 0, 1, 1)self.pushButton_save = QPushButton('save', self)self.pushButton_save.setEnabled(False)self.gridLayout.addWidget(self.pushButton_save, 1, 0, 1, 1)spacerItem = QSpacerItem(20, 549, QSizePolicy.Minimum, QSizePolicy.Expanding)self.gridLayout.addItem(spacerItem, 2, 0, 1, 1)self.graphicsView = GraphicsView(self.picture, self)self.graphicsView.setVerticalScrollBarPolicy(Qt.ScrollBarAlwaysOff)self.graphicsView.setHorizontalScrollBarPolicy(Qt.ScrollBarAlwaysOff)self.gridLayout.addWidget(self.graphicsView, 0, 1, 3, 1)def pushButton_cut_clicked(self):if self.graphicsView.image_item.is_start_cut:self.graphicsView.image_item.is_start_cut = Falseself.graphicsView.image_item.setCursor(Qt.ArrowCursor)  # 箭头光标else:self.graphicsView.image_item.is_start_cut = Trueself.graphicsView.image_item.setCursor(Qt.CrossCursor)  # 十字光标def pushButton_save_clicked(self):rect = QRect(self.graphicsView.image_item.start_point.toPoint(),self.graphicsView.image_item.end_point.toPoint())new_pixmap = self.graphicsView.image_item.pixmap().copy(rect)new_pixmap.save(r'test.png')if __name__ == '__main__':import sysapp = QApplication(sys.argv)form = Form()form.show()app.exec_()

重构QGraphicsView和QGraphicsPixmapItem

from PyQt5.QtCore import QRectF, Qt, pyqtSignal
from PyQt5.QtGui import QColor, QPixmap, QPen
from PyQt5.QtWidgets import QGraphicsView, QGraphicsPixmapItem, QGraphicsScene, QGraphicsItemclass GraphicsView(QGraphicsView):save_signal = pyqtSignal(bool)def __init__(self, picture, parent=None):super(GraphicsView, self).__init__(parent)# 设置放大缩小时跟随鼠标self.setTransformationAnchor(QGraphicsView.AnchorUnderMouse)self.setResizeAnchor(QGraphicsView.AnchorUnderMouse)self.scene = QGraphicsScene()self.setScene(self.scene)self.image_item = GraphicsPixmapItem(QPixmap(picture))self.image_item.setFlag(QGraphicsItem.ItemIsMovable)self.scene.addItem(self.image_item)size = self.image_item.pixmap().size()# 调整图片在中间self.image_item.setPos(-size.width() / 2, -size.height() / 2)self.scale(0.1, 0.1)def wheelEvent(self, event):'''滚轮事件'''zoomInFactor = 1.25zoomOutFactor = 1 / zoomInFactorif event.angleDelta().y() > 0:zoomFactor = zoomInFactorelse:zoomFactor = zoomOutFactorself.scale(zoomFactor, zoomFactor)def mouseReleaseEvent(self, event):'''鼠标释放事件'''# print(self.image_item.is_finish_cut, self.image_item.is_start_cut)if self.image_item.is_finish_cut:self.save_signal.emit(True)else:self.save_signal.emit(False)class GraphicsPixmapItem(QGraphicsPixmapItem):save_signal = pyqtSignal(bool)def __init__(self, picture, parent=None):super(GraphicsPixmapItem, self).__init__(parent)self.setPixmap(picture)self.is_start_cut = Falseself.current_point = Noneself.is_finish_cut = Falsedef mouseMoveEvent(self, event):'''鼠标移动事件'''self.current_point = event.pos()if not self.is_start_cut or self.is_midbutton:self.moveBy(self.current_point.x() - self.start_point.x(),self.current_point.y() - self.start_point.y())self.is_finish_cut = Falseself.update()def mousePressEvent(self, event):'''鼠标按压事件'''super(GraphicsPixmapItem, self).mousePressEvent(event)self.start_point = event.pos()self.current_point = Noneself.is_finish_cut = Falseif event.button() == Qt.MidButton:self.is_midbutton = Trueself.update()else:self.is_midbutton = Falseself.update()def paint(self, painter, QStyleOptionGraphicsItem, QWidget):super(GraphicsPixmapItem, self).paint(painter, QStyleOptionGraphicsItem, QWidget)if self.is_start_cut and not self.is_midbutton:# print(self.start_point, self.current_point)pen = QPen(Qt.DashLine)pen.setColor(QColor(0, 150, 0, 70))pen.setWidth(3)painter.setPen(pen)painter.setBrush(QColor(0, 0, 255, 70))if not self.current_point:returnpainter.drawRect(QRectF(self.start_point, self.current_point))self.end_point = self.current_pointself.is_finish_cut = True

成果

这篇关于QGraphicsView 实现图片裁剪功能的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

使用opencv优化图片(画面变清晰)

文章目录 需求影响照片清晰度的因素 实现降噪测试代码 锐化空间锐化Unsharp Masking频率域锐化对比测试 对比度增强常用算法对比测试 需求 对图像进行优化,使其看起来更清晰,同时保持尺寸不变,通常涉及到图像处理技术如锐化、降噪、对比度增强等 影响照片清晰度的因素 影响照片清晰度的因素有很多,主要可以从以下几个方面来分析 1. 拍摄设备 相机传感器:相机传

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

让树莓派智能语音助手实现定时提醒功能

最初的时候是想直接在rasa 的chatbot上实现,因为rasa本身是带有remindschedule模块的。不过经过一番折腾后,忽然发现,chatbot上实现的定时,语音助手不一定会有响应。因为,我目前语音助手的代码设置了长时间无应答会结束对话,这样一来,chatbot定时提醒的触发就不会被语音助手获悉。那怎么让语音助手也具有定时提醒功能呢? 我最后选择的方法是用threading.Time

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

C#实战|大乐透选号器[6]:实现实时显示已选择的红蓝球数量

哈喽,你好啊,我是雷工。 关于大乐透选号器在前面已经记录了5篇笔记,这是第6篇; 接下来实现实时显示当前选中红球数量,蓝球数量; 以下为练习笔记。 01 效果演示 当选择和取消选择红球或蓝球时,在对应的位置显示实时已选择的红球、蓝球的数量; 02 标签名称 分别设置Label标签名称为:lblRedCount、lblBlueCount

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略 1. 特权模式限制2. 宿主机资源隔离3. 用户和组管理4. 权限提升控制5. SELinux配置 💖The Begin💖点点关注,收藏不迷路💖 Kubernetes的PodSecurityPolicy(PSP)是一个关键的安全特性,它在Pod创建之前实施安全策略,确保P

工厂ERP管理系统实现源码(JAVA)

工厂进销存管理系统是一个集采购管理、仓库管理、生产管理和销售管理于一体的综合解决方案。该系统旨在帮助企业优化流程、提高效率、降低成本,并实时掌握各环节的运营状况。 在采购管理方面,系统能够处理采购订单、供应商管理和采购入库等流程,确保采购过程的透明和高效。仓库管理方面,实现库存的精准管理,包括入库、出库、盘点等操作,确保库存数据的准确性和实时性。 生产管理模块则涵盖了生产计划制定、物料需求计划、