使用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 相关文章推荐
Python3 正在毁灭 Python的原因分析
Nov 28 Python
Python中使用第三方库xlutils来追加写入Excel文件示例
Apr 05 Python
Python第三方库xlrd/xlwt的安装与读写Excel表格
Jan 21 Python
Python实现读取txt文件并画三维图简单代码示例
Dec 09 Python
linux查找当前python解释器的位置方法
Feb 20 Python
Python3实现的旋转矩阵图像算法示例
Apr 03 Python
Python发展史及网络爬虫
Jun 19 Python
python生成大写32位uuid代码
Mar 03 Python
PHP基于phpqrcode类库生成二维码过程解析
May 28 Python
使用python编写一个语音朗读闹钟功能的示例代码
Jul 14 Python
selenium携带cookies模拟登陆CSDN的实现
Jan 19 Python
详解Python中下划线的5种含义
Jul 15 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中json_encode、json_decode与serialize、unserialize的性能测试分析
2010/06/09 PHP
解决PHP在DOS命令行下却无法链接MySQL的技术笔记
2010/12/29 PHP
使用PHPUnit进行单元测试并生成代码覆盖率报告的方法
2019/03/08 PHP
为你的 Laravel 验证器加上多验证场景的实现
2020/04/07 PHP
扩展easyui.datagrid,添加数据loading遮罩效果代码
2010/11/02 Javascript
JQuery实现表格中相同单元格合并示例代码
2013/06/26 Javascript
jQuery zclip插件实现跨浏览器复制功能
2015/11/02 Javascript
Markdown+Bootstrap图片自适应属性详解
2016/05/21 Javascript
Sortable.js拖拽排序使用方法解析
2016/11/04 Javascript
node.js平台下的mysql数据库配置及连接
2017/03/31 Javascript
vue实现全选和反选功能
2017/08/31 Javascript
基于js的变量提升和函数提升(详解)
2017/09/17 Javascript
js中bool值的转换及“&&”、“||”、 “!!”详解
2017/12/21 Javascript
详解Vue This$Store总结
2018/12/17 Javascript
微信小程序中使用Async-await方法异步请求变为同步请求方法
2019/03/28 Javascript
CountUp.js数字滚动插件使用方法详解
2019/10/17 Javascript
JS实现导航栏楼层特效
2020/01/01 Javascript
vue通过v-html指令渲染的富文本无法修改样式的解决方案
2020/05/20 Javascript
python采用django框架实现支付宝即时到帐接口
2016/05/17 Python
Python实现SSH远程登陆,并执行命令的方法(分享)
2017/05/08 Python
Python读取数据集并消除数据中的空行方法
2018/07/12 Python
python实现转圈打印矩阵
2019/03/02 Python
浅谈Python 递归算法指归
2019/08/22 Python
Python 无限级分类树状结构生成算法的实现
2021/01/21 Python
浅析与CSS3的loading动画加载相关的transition优化
2015/05/18 HTML / CSS
英国领先的在线礼品店:Getting Personal
2019/09/24 全球购物
甜美蛋糕店创业计划书
2014/01/30 职场文书
学生周末回家住宿长期请假条
2014/02/15 职场文书
英文请假条
2014/04/11 职场文书
乡镇党建工作汇报材料
2014/08/14 职场文书
高中美术教师事迹材料
2014/08/22 职场文书
授权委托书(完整版)
2014/09/10 职场文书
社区安全温馨提示语
2015/07/14 职场文书
详解前端任务构建利器Gulp.js使用指南
2021/04/30 Javascript
教你怎么用Python生成九宫格照片
2021/05/20 Python
浅谈Redis的事件驱动模型
2022/05/30 Redis