使用PyQt的QLabel组件实现选定目标框功能的方法示例


Posted in Python onMay 19, 2020

问题背景

  基于PyQt5开发了一个可以用于目标跟踪的软件,在开发过程中遇到一个问题,就是如何在PyQt5的组件QLable中自主选定目标框,这个在opencv里面有专门的函数完成这个工作:cv2.selectROI(),我的目的就是在QLabel的基础上,实现类似函数cv2.selectROI()的功能,这样在运行程序的过程中,就能在视频框里面直接选取感兴趣区域。直接贴出实现的最终效果:

使用PyQt的QLabel组件实现选定目标框功能的方法示例

上图中的红色框框就是在QLabel的基础上实现的功能。

实现思路

  具体要实现的功能是,在视频显示区域,点击鼠标左键,开启选择,按照鼠标左键,移动游标,慢慢地绘制出红色的目标框。释放鼠标左键就停止选择目标框。最开始以为PyQt好歹也会提供这样的类来进行开发吧,后来发现其实是没有的,没办法只能写一个QLabel类的子类了。子类的命名为Label,继承自QLabel类,在子类中重写鼠标事件函数,接受鼠标在Label对象上位置信号。PyQt本来就有自己的事件循环,当鼠标落在视频显示区域的时候,触发到Label的鼠标事件,那么就可以开始绘制目标框了。

  这里要记录的就是鼠标按下左键时候的起始坐标pos_1和移动坐标pos_2pos_1=(x0,y0),pos_2=(x1,y1)

  重写按下鼠标事件 按下鼠标左键,触发事件函数mousePressEvent(),事件函数打开绘制标志位self.select_roi_flag,传入事件对象数据,初始化起始坐标x0,y0

  重写释放鼠标事件 按下鼠标左键,触发事件函数mousePressEvent(),关闭绘制标志位self.select_roi_flag

  绘制事件 继承鼠标事件绘制类,创建画笔类对象,在这可以设置画笔的颜色,画线的粗细,如果绘制标志位self.select_roi_flag是打开的,那么将事件对象的位置数据传给x1,y1QRect类是是PyQt的内置数据结构,具体结构是这样的Rect=(x,y,w,h),之后就调用画笔对象方法动态绘制目标框。直到绘制标志位被关闭,就是释放鼠标,则停止绘画。

具体实现代码:

from PyQt5.QtWidgets import QLabel
from PyQt5.QtCore import Qt,QRect
from PyQt5.QtGui import QPainter,QPen

class Label(QLabel):
  x0=0
  y0=0
  x1=0
  y1=0
  open_mouse_flag=False
  select_roi_flag=False
  draw_roi_flag=False
  clear_flag=False
  rect = QRect()

  #按下鼠标
  def mousePressEvent(self, event):
    if self.open_mouse_flag is True:
      self.select_roi_flag=True
      self.x0=event.x()
      self.y0=event.y()

  #释放鼠标
  def mouseReleaseEvent(self, event):
    self.select_roi_flag=False

  #移动鼠标
  def mouseMoveEvent(self, event):
    if self.select_roi_flag is True:
      self.x1=event.x()
      self.y1=event.y()
      if self.draw_roi_flag is True:
        self.update()

  #绘制事件
  def paintEvent(self,event):
    super().paintEvent(event)
    painter = QPainter(self)
    painter.setPen(QPen(Qt.red, 5, Qt.SolidLine))
    if self.clear_flag is True:
      self.x0=0
      self.y0=0
      self.x1=0
      self.y1=0
    self.rect = QRect(self.x0, self.y0, abs(self.x1 - self.x0), abs(self.y1 - self.y0))
    painter.drawRect(self.rect)
    self.update()

其他要注意的问题

  子类Label除了能自定义选择目标框,还要在更新内容是清除绘制内容,实现这个功能可以通过设置清空标志位clear_flag,当标志位打开的时候,将起始坐标和更新坐标重置为:(0,0)(0,0),这样绘制内容就被更新了。
具体实现代码:

# 清除label对象的绘制内容
def clear_label(self):
  self.label_show.clear_flag = True
  self.label_show.clear()

  此外我还重写了键盘事件,通过敲击键盘来控制鼠标的绘制事件,这里的内容主要包括切换游标,开启绘制事件,确认绘制事件。

具体实现代码:

# 重写键盘事件
def keyPressEvent(self, QKeyEvent):
  if self.open_keyboard_flag is True:         # 当键盘事件为真的是才有键盘事件监控
    if QKeyEvent.key() == Qt.Key_S:
      self.label_show.setCursor(Qt.CrossCursor)  # 切换游标为十字型
      self.label_show.open_mouse_flag = True
      self.label_show.draw_roi_flag = True
    if QKeyEvent.key() == Qt.Key_Q:         # 按下'q'键键盘监控关闭
      self.label_show.unsetCursor()
      self.label_show.draw_roi_flag = False
      self.label_show.open_mouse_flag = False
      self.open_keyboard_flag = False

到此这篇关于使用PyQt的QLabel组件实现选定目标框功能的方法示例的文章就介绍到这了,更多相关PyQt QLabel选定目标框 内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Python 相关文章推荐
详解Python中用于计算指数的exp()方法
May 14 Python
浅谈django的render函数的参数问题
Oct 16 Python
pandas DataFrame 删除重复的行的实现方法
Jan 29 Python
pyqt5 从本地选择图片 并显示在label上的实例
Jun 13 Python
python数据预处理之数据标准化的几种处理方式
Jul 17 Python
详解python实现小波变换的一个简单例子
Jul 18 Python
使用OpenCV实现仿射变换—缩放功能
Aug 29 Python
python 解决print数组/矩阵无法完整输出的问题
Feb 19 Python
python如何运行js语句
Sep 09 Python
基于Python的接口自动化unittest测试框架和ddt数据驱动详解
Jan 27 Python
浅谈Python实现opencv之图片色素的数值运算和逻辑运算
Jun 23 Python
如何利用python创作字符画
Jun 25 Python
python 数据分析实现长宽格式的转换
May 18 #Python
如何把外网python虚拟环境迁移到内网
May 18 #Python
python 实现 hive中类似 lateral view explode的功能示例
May 18 #Python
pandas dataframe 中的explode函数用法详解
May 18 #Python
Python pandas 列转行操作详解(类似hive中explode方法)
May 18 #Python
Win 10下Anaconda虚拟环境的教程
May 18 #Python
python异常处理之try finally不报错的原因
May 18 #Python
You might like
php简单的留言板与回复功能具体实现
2014/02/19 PHP
php堆排序实现原理与应用方法
2015/01/03 PHP
[原创]php token使用与验证示例【测试可用】
2017/08/30 PHP
Javascript select控件操作大全(新增、修改、删除、选中、清空、判断存在等)
2008/12/19 Javascript
JavaScript中的alert()函数使用技巧详解
2014/12/29 Javascript
javascript实现禁止鼠标滚轮事件
2015/07/24 Javascript
11种ASP连接数据库的方法
2015/09/18 Javascript
JS实现可自定义大小,可双击关闭的弹出层效果
2015/10/16 Javascript
jQuery链式操作实例分析
2015/11/16 Javascript
原生js实现自由拖拽弹窗代码demo
2016/06/29 Javascript
ionic组件ion-tabs选项卡切换效果实例
2016/08/27 Javascript
AngularJS 中的Promise --- $q服务详解
2016/09/14 Javascript
JS前端笔试题分析
2016/12/19 Javascript
原生JS下拉加载插件分享
2016/12/26 Javascript
Easyui在treegrid添加控件的实现方法
2017/06/23 Javascript
如何通过非数字与字符的方式实现PHP WebShell详解
2017/07/02 Javascript
layui table 表格模板按钮的实例代码
2019/09/21 Javascript
基于JS+HTML实现弹窗提示是否确认提交功能
2020/06/17 Javascript
vue实现广告栏上下滚动效果
2020/11/26 Vue.js
[00:52]玛尔斯技能全介绍
2019/03/06 DOTA
[01:38:19]夜魇凡尔赛茶话会 第五期
2021/03/11 DOTA
python Django连接MySQL数据库做增删改查
2013/11/07 Python
PyQt5 窗口切换与自定义对话框的实例
2019/06/20 Python
python实现电子词典
2020/03/03 Python
简单了解django处理跨域请求最佳解决方案
2020/03/25 Python
台湾前三大B2C购物网站:MOMO购物网
2017/04/27 全球购物
便利店投资创业计划书
2014/02/08 职场文书
《四季》教学反思
2014/04/08 职场文书
社区党的群众路线教育实践活动总结材料
2014/10/31 职场文书
医院保洁员岗位职责
2015/02/13 职场文书
2015年度环卫处工作总结
2015/07/24 职场文书
车辆管理制度范本
2015/08/05 职场文书
采购员工作总结范文
2015/08/12 职场文书
某学校的2019年度工作报告范本
2019/10/11 职场文书
springboot集成flyway自动创表的详细配置
2021/06/26 Java/Android
SQL SERVER实现连接与合并查询
2022/02/24 SQL Server