PySide(PyQT)使用场景(QGraphicsScene)进行动态标注的一个demo

devtools/2025/2/10 21:40:44/

 用以标注图像的一个基本框架demo

python">import sys
from PySide6.QtWidgets import QApplication, QGraphicsView, QGraphicsScene, QMainWindow, QLabel, QGraphicsPixmapItem
from PySide6.QtGui import QPixmap, QPainter, QTransform
from PySide6.QtCore import Qt, QPointF, Slot, Signalclass ImageViewer(QGraphicsView):mouse_pos = Signal(int, int)  # 原图像素坐标信号def __init__(self, image_path, parent=None):super(ImageViewer, self).__init__(parent)# 场景的初始化def init_scene():# 设置场景self.scene = QGraphicsScene(self)self.setScene(self.scene)# 设置渲染提示self.setRenderHint(QPainter.Antialiasing)   # 开启抗锯齿self.setRenderHint(QPainter.SmoothPixmapTransform)  # 开启平滑缩放# 设置缩放锚点self.setTransformationAnchor(QGraphicsView.AnchorUnderMouse)  # 转换时以鼠标为中心self.setResizeAnchor(QGraphicsView.AnchorUnderMouse)  # 缩放时以鼠标为中心# 创建基础的显示内容pixmap = QPixmap(image_path)self.pixmap_item = QGraphicsPixmapItem(pixmap)self.scene.addItem(self.pixmap_item)# 标记点的数量self.mark_count = 0# 遮罩的数量self.mask_count = 0init_scene()self.mouse_pos.connect(self.update_mouse_position)def add_mark(self, pixmap, pos=(0, 0), scale=1.0):"""添加标记到场景:param pixmap: 图像数据:param pos:  坐标:param scale: 缩放比例:return:  None"""# 加载图像pixmap_item = QGraphicsPixmapItem(pixmap)  # 创建图形项mark_x = pos[0] - pixmap.width() / 2 * scalemark_y = pos[1] - pixmap.height() / 2 * scalepixmap_item.setPos(mark_x, mark_y)  # 设置位置if scale != 1.0:  # 设置比例pixmap_item.setScale(scale)self.scene.addItem(pixmap_item)self.mark_count += 1def add_mask(self):"""添加遮罩:return:"""passdef remove_mark(self):"""删除一个标记:return: None"""if self.mark_count > 0:self.scene.removeItem(self.scene.items()[0])self.mark_count -= 1def remove_mask(self):"""删除一个遮罩:return: None"""if self.mask_count > 0:self.scene.removeItem(self.scene.items()[0])self.mask_count -= 1def remove_all_marks(self):"""删除所有标记:return:"""while self.mark_count > 0:self.scene.removeItem(self.scene.items()[0])self.mark_count -= 1# # 使用变换矩阵(如果需要)def transform(self, t):# self.transform = t# self.pixmap_item.setTransform(self.transform)pass# 处理滚轮事件以实现缩放def wheelEvent(self, event):factor = 1.001 ** event.angleDelta().y()  # 滚轮每滚动一格,缩放比例变化self.scale(factor, factor)# 处理鼠标单击事件以显示原图像素坐标def mousePressEvent(self, event):super(ImageViewer, self).mousePressEvent(event)scene_pos = self.mapToScene(event.position().toPoint())  # 记录当前鼠标在场景中的位置pixmap_rect = self.pixmap_item.boundingRect()   # 记录图像的边界矩形# 如果鼠标在图像边界矩形内,则显示像素坐标if pixmap_rect.contains(scene_pos):pos = self.pixmap_item.mapFromScene(scene_pos)  # 计算图像坐标x, y = pos.x(), pos.y()print(f"鼠标在图像内,坐标为: ({x}, {y})")if event.button() == Qt.LeftButton:  # 左键if event.modifiers() == Qt.ShiftModifier:  # Shift 键self.remove_mark()    # 删除标记else:self.add_mark(pixmap_mark, (x, y), 0.6)  # 添加标记elif event.button() == Qt.RightButton:   # 右键if event.modifiers() == Qt.ShiftModifier:   # Shift 键self.remove_all_marks()   # 删除所有标记else:self.add_mark(pixmap_unmark, (x, y), 0.6)   # 添加标记@Slot()def update_mouse_position(self, x, y):pass# 更新状态栏显示# self.status_bar.showMessage(f"原图像素坐标: ({x}, {y})")class MainWindow(QMainWindow):def __init__(self, image_path):super(MainWindow, self).__init__()self.setGeometry(100, 100, 800, 600)# 创建状态栏self.status_bar = self.statusBar()# 创建 ImageViewer 实例self.image_viewer = ImageViewer(image_path, self)self.setCentralWidget(self.image_viewer)if __name__ == "__main__":app = QApplication(sys.argv)# 生成标记图像pixmap_mark = QPixmap("mark.png")   # 标记图像(前景)pixmap_unmark = QPixmap("un_mark.png")   # 未标记图像(背景)image_path = "IMG_PP.jpg"      # 基础图像路径# 创建主窗口window = MainWindow(image_path)window.show()sys.exit(app.exec())

截图:


http://www.ppmy.cn/devtools/157741.html

相关文章

c/c++蓝桥杯经典编程题100道(16)链表反转

链表反转 c/c蓝桥杯经典编程题100道-目录-CSDN博客 目录 链表反转 一、题型解释 二、例题问题描述 三、C语言实现 解法1:迭代反转(难度★) 解法2:递归反转(难度★★) 解法3:分组反转&am…

Excel大数据量导入导出

github源码 地址(更详细) : https://github.com/alibaba/easyexcel 文档:读Excel(文档已经迁移) B 站视频 : https://www.bilibili.com/video/BV1Ff4y1U7Qc 一、JAVA解析EXCEL工具EasyExcel Java解析、生成Excel比较…

可以在个人电脑上部署的主流开源大模型

目前主流开源的大模型发展迅速,许多模型经过优化后可以在个人电脑(甚至CPU或消费级GPU)上运行。以下是当前主流的开源大模型及其在个人设备上的部署可行性总结: 一、主流开源大模型 1.DeepSeek系列 DeepSeek大语言模型算法:以Transformer架构为基础,自主研发的深度神经网…

问卷数据分析|SPSS之分类变量描述性统计

1.点击分析--描述统计--频率 2. 选中分类变量,点击中间箭头 3.图表选中条形图,图表值选择百分比,选择确定 4.这里显示出了描述性统计的结果 5.下面就是图形,但SPSS画的图形都不是很好啊看,建议用其他软件画图&#xff…

Bash (Bourne-Again Shell)、Zsh (Z Shell)

文章目录 1. 历史背景2. 主要区别3. 功能对比自动补全插件和主题路径扩展提示符定制 4. 性能5. 使用场景6. 如何切换 Shell7. 总结 以下是 Bash 和 Zsh 之间的主要区别,列成表格方便对比: 特性BashZsh默认Shell大多数Linux发行版默认ShellmacOS默认She…

通过docker安装部署deepseek以及python实现

前提条件 Docker 安装:确保你的系统已经安装并正确配置了 Docker。可以通过运行 docker --version 来验证 Docker 是否安装成功。 网络环境:保证设备有稳定的网络连接,以便拉取 Docker 镜像和模型文件。 步骤一:拉取 Ollama Docker 镜像 Ollama 可以帮助我们更方便地管理…

Android开发签名校验

Android开发签名校验 有一些平台需要我们做签名校验才能通过审核,其实做Android签名校验也不是很难 直接上代码: class SignCheck {/*** 获取应用的签名*/private fun getCer(mContext: Context): String? {var packageInfo: PackageInfo? nulltry…

[权限提升] Linux 提权 维持 — 系统错误配置提权 - 通配符(ws)注入提权

关注这个专栏的其他相关笔记:[内网安全] 内网渗透 - 学习手册-CSDN博客 0x01:通配符(ws)注入提权原理 通配符注入提权的核心是利用通配符的扩展特性,在命令执行时生成意外的参数或文件名,从而改变命令的行…