一、写在前面

作为一名测试,有时候经常会遇到需要录屏记录自己操作,方便后续开发同学定位。以前都是用ScreenToGif来录屏制作成动态图,偶尔的机会看到python也能实现。那就赶紧学习下。

二、效果展示

三、知识串讲

这次要讲的东西可能比较多了,涉及到pyqt5 GUI软件的制作、QThread多线程的使用、Sikuli库的图形操作、win32库的模拟键盘操作、cv2库的写视频文件等。下面我们一点点来蚕食我这次写的代码。

1、GUI界面制作

这次我用的是现成的Pyqt5界面布局类,QVBoxLayout。这个类可以快速协助我完成按钮的垂直分布,而且按钮添加也更方便。

button1 = QPushButton("自定义录屏")
layout.addWidget(button1)

两行代码就完成了按钮的命名和添加。我之前玩qt时,用的都是qt的UI界面,对应生成的组件代码也比较复杂。因此,在开发一些少量按钮、简单布局时可以用QVBoxLayout类。如果喜欢水平布局,可以用QHBoxLayout类,使用方法是一样的。

另外,在按钮点击关联的功能函数,即work()方法时,如果想带参数,可以通过lambda匿名函数来实现。这 也是个小技巧。

# 不带参数
button1.clicked.connect(self.work) # 带参数
button1.clicked.connect(lambda: self.work(1))

2、QThread类的多线程使用

因为录屏工具有开始和停止两个功能,一开始时我用的是单线程,发现工具就会卡死。查了一些资料,发现针对这种情况,应该要使用多线程来实现,而QT库中本身就有多线程类–QThread。

使用方法是通过继承QThread类,重写run方法来实现的。

(但是其实这种使用方法,QT大神们是不赞成这样使用的,我会在第2篇文章中再简单说明更好的多线程使用方法)

这 里要注意,work()函数必须是Ui_Mainwindow类方法,因为如果不是类方法,会在运行GUI时导致生命周期直接结束,导致录屏代码没见运行就报错退出。

class WorkThread(QThread): def __init__(self, n):super(WorkThread, self).__init__()self.n = n def run(self):XXXXX

3、sikuli库图形识别

由于这个库的使用方法和介绍,我在之前的博客里已经提过 了。因此只简单地呈现下代码。这段代码主要是为了自定义录屏时,可以获取选择范围的坐标值,并传值给recording函数,从而完成自定义录屏功能。

def SelectRegion():jvmPath = jpype.get_default_jvm_path()jpype.startJVM(jvmPath, '-ea', '-Djava.class.path=F:\\sikuli\\1\\sikulixapi.jar') #加载jar包路径Screen = jpype.JClass('org.sikuli.script.Screen')myscreen = Screen()region = myscreen.selectRegion() # 自定义获取屏幕范围return region

4、win32库模拟键盘操作

其实这个库不用也是可以的,我为什么要用呢?主要是为了方便用户在进行录屏时,能自动将工具界面缩小。一切为了用户嘛!

以下这段代码 是为了缩小工具窗口,其中91表示左win键,40表示方向向下键。****即win+向下键是可以实现窗口缩小功能的。****keybd_event(91, 0, 0, 0)表示按下win键,

keybd_event(91, 0, win32con.KEYEVENTF_KEYUP, 0)则是松开win键。

另外,这里为什么要加 上sleep(0.5)?这是因为在按下win键后要延迟按方向键,不然是 不起作用的。

def Minimize_Window():win32api.keybd_event(91, 0, 0, 0)time.sleep(0.5)win32api.keybd_event(40, 0, 0, 0)time.sleep(0.5)win32api.keybd_event(91, 0, win32con.KEYEVENTF_KEYUP, 0)win32api.keybd_event(40, 0, win32con.KEYEVENTF_KEYUP, 0)

5、录屏主代码

这段代码其实网上已经有很多类似的代码,并且我已经加了注释,相信大家应该能理解。这里我想注明下的是:如何停止录屏。

如果大家有去 网上查如何停止录屏的方法,很多人都会写以下代码:

if cv2.waitKey(1) & 0xFF == ord('q'): break

然后告诉你,按q键就会停止录屏。但是你会发现,实际情况根本停止不了,为什么呢?因为还 有一句屏幕显示的代码:

cv2.imshow('imm', img_bgr) if cv2.waitKey(1) & 0xFF == ord('q'): break

如果你不亲自执行一次,你以为会万事大吉,但你错了。这样写,会导致你的电脑屏幕被每一帧画面给撑暴!因为用的while True,因此每一帧画面都会显示,即1S 25帧画面会不停地显示在你桌面上!

因此,综上的问题,我采用了一种取巧的方法:在录屏开始时生成一个标记文件,通过标记文件是否被删除来判断是否要停止录屏功能。

四、示例代码

1、工具GUI界面代码:

# coding=utf-8 # @Software : PyCharm #Python学习群855408893import sys from PyQt5.QtCore import *
from PyQt5.QtWidgets import *
import time import win32api,win32con from recording import *class WorkThread(QThread): def __init__(self, n):super(WorkThread, self).__init__()self.n = n def run(self): if self.n == 1:Minimize_Window()Recording(1) elif self.n == 2:Minimize_Window()Recording(2) else:StopRecording() def Minimize_Window():win32api.keybd_event(91, 0, 0, 0)time.sleep(0.5)win32api.keybd_event(40, 0, 0, 0)time.sleep(0.5)win32api.keybd_event(91, 0, win32con.KEYEVENTF_KEYUP, 0)win32api.keybd_event(40, 0, win32con.KEYEVENTF_KEYUP, 0) class Ui_Mainwindow(): def setupUi(self, top): # 垂直布局类QVBoxLayoutlayout = QVBoxLayout(top) # 添加录屏相关按钮button1 = QPushButton("自定义录屏")layout.addWidget(button1)button2 = QPushButton("全屏录屏")layout.addWidget(button2)button3 = QPushButton("停止录屏")layout.addWidget(button3)self.text = QPlainTextEdit('欢迎使用!')layout.addWidget(self.text)button1.clicked.connect(lambda: self.work(1))button2.clicked.connect(lambda: self.work(2))button3.clicked.connect(lambda: self.work(3)) def work(self, n): if n == 1 : print('已选择自定义录屏:')self.text.setPlainText('正在录屏中,请等待……') elif n == 2 : print('已选择全屏录屏:')self.text.setPlainText('正在录屏中,请等待……') else: print('已选择结束录屏:')self.text.setPlainText('录屏结束!(点击关闭按钮,可退出程序!)')self.workThread = WorkThread(n)self.workThread.start() if __name__ == "__main__":app = QApplication(sys.argv)top = QWidget()top.setWindowTitle('录屏小工具')top.resize(300, 170)ui = Ui_Mainwindow()ui.setupUi(top)top.show()sys.exit(app.exec_())# coding=utf-8

2、录屏函数

在学习过程中有什么不懂得可以加我的
python学习qun,855408893
群里有不错的学习视频教程、开发工具与电子书籍。
与你分享python企业当下人才需求及怎么从零基础学习好python,和学习什么内容# coding=utf-8 # @Software : PyCharmfrom PIL import ImageGrab import numpy as np import cv2 import os import jpype def Recording(tag=1): # 录屏开始时创建test.txt,作为结束录屏的条件#Python学习群855408893if not os.path.exists('test.txt'):f = open('test.txt', 'w')f.close() # 根据tag值判断自定义录屏或全录屏if tag == 1:r = SelectRegion()record_region = (r.x, r.y, r.w + r.x, r.h + r.y) # 自定义录屏的范围(左上坐标、右下坐标)elif tag == 2:record_region = Noneimage = ImageGrab.grab(record_region)  # 获取指定范围的屏幕对象width, height = image.sizefourcc = cv2.VideoWriter_fourcc(*'XVID')video = cv2.VideoWriter('test.avi', fourcc, 25, (width, height)) # 默认视频为25帧while True:captureImage = ImageGrab.grab(record_region)  # 抓取指定范围的屏幕frame = cv2.cvtColor(np.array(captureImage), cv2.COLOR_RGB2BGR)video.write(frame) # 将每帧画面写视频文件# 停止录屏的条件:test.txt被删除if not os.path.exists('test.txt'): break video.release()cv2.destroyAllWindows() def SelectRegion():jvmPath = jpype.get_default_jvm_path()jpype.startJVM(jvmPath, '-ea', '-Djava.class.path=F:\\sikuli\\1\\sikulixapi.jar') #加载jar包路径Screen = jpype.JClass('org.sikuli.script.Screen')myscreen = Screen()region = myscreen.selectRegion() # 自定义获取屏幕范围return region def StopRecording():os.remove('test.txt') #停止录屏的触发条件if __name__ == "__main__":Recording()

五、总结

至此,基本实现了录屏小工具的代码开发。但是如果你是对代码中的相关库不熟悉,或者都没下载相关的库,那我相信你还会遇到很多坑。因此,为了方便一些小伙伴能快速把代码跑起来,我将在下一篇文章中讲讲我在开发时遇到的一些坑,方便大家能避免这些问题。好了,今天就先到这里!Bye!

【趣味案例】用Python来做一个屏幕录制工具相关推荐

  1. python屏幕录制与回放_使用Python来做一个屏幕录制工具的操作代码

    本文给大家分享使用Python来做一个屏幕录制工具,通过实例代码给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下 一.写在前面 作为一名测试,有时候经常会遇到需要录屏记录自己操作,方 ...

  2. 脚本录制软件python 按键精灵 tc_使用Python来做一个屏幕录制工具的操作代码

    一.写在前面 作为一名测试,有时候经常会遇到需要录屏记录自己操作,方便后续开发同学定位.以前都是用ScreenToGif来录屏制作成动态图,偶尔的机会看到python也能实现.那就赶紧学习下. 二.效 ...

  3. python minimize_【趣味案例】用Python来做一个屏幕录制工具

    一.写在前面 作为一名测试,有时候经常会遇到需要录屏记录自己操作,方便后续开发同学定位.以前都是用ScreenToGif来录屏制作成动态图,偶尔的机会看到python也能实现.那就赶紧学习下. 二.效 ...

  4. python实现屏幕视频录制_用Python来做一个屏幕录制工具

    一.写在前面 作为一名测试,有时候经常会遇到需要录屏记录自己操作,方便后续开发同学定位.以前都是用screentogif来录屏制作成动态图,偶尔的机会看到python也能实现.那就赶紧学习下. 二.效 ...

  5. 用 Python 来做一个屏幕录制工具

    前言 女朋友是一个老师,在工作中经常会遇到需要录屏记录自己操作,方便后续开发同学定位.因为录屏软件动不动就开始收费,所以她经常更换录屏软件.闲暇之余,我就觉得手痒,感觉可以用万能的 Python 来解 ...

  6. 绿色节能,怎么用ESP32 + HaaS Python怎么做一个起夜小灯?接上个“智慧路灯”案例中的留下的地下室人体感应亮灭灯

    自上个案例 "用ESP32 + HaaS Python做一个 智慧路灯 系统,稍加改造也能用在地下室哦"发布以来,数度冲上CSDN 领域热度排行榜 与 全站热度排行榜,也有朋友与我 ...

  7. python实战扫码下载_实例:用 Python 做一个扫码工具

    原标题:实例:用 Python 做一个扫码工具 来自公众号: 新建文件夹X 链接:https://blog.csdn.net/ZackSock/article/details/108610957Pyt ...

  8. python制作聊天机器人原理_用 Python 来做一个聊天机器人吧!(一)

    在我的一个回答里,我提到了用 Python 搭建聊天机器人.从今天开始,我就带着大家从0开始搭建一个聊天机器人. (顺便说一句,我喜欢把链接像上面这样加在文字里,如果找不到文中所说的资源,可以看看周围 ...

  9. 用Python做一个安全攻防工具:端口嗅探器(9)

    传送门 本系列原创博文传送门: 用Python做一个安全攻防工具:端口嗅探器(1) 用Python做一个安全攻防工具:端口嗅探器(2) 用Python做一个安全攻防工具:端口嗅探器(3) 用Pytho ...

最新文章

  1. Shouldn't CSRs automatically add default version?
  2. WinAPI: GetWindow - 获取与指定窗口具有指定关系的窗口的句柄
  3. Silverlight WCF RIA服务(二十七)Silverlight 客户端 8
  4. azure webjob java_使用 WebJobs 运行后台任务 | Azure Docs
  5. 交叉渡线道岔规格_交叉渡线道岔选择基本原则
  6. 双屏怎么快速切换鼠标_双屏触摸一体机功能特点解析
  7. 微信公众平台开发视频教程
  8. 基于微信小程序的点餐系统设计
  9. 读书笔记:调试软件 张银奎
  10. CS 61A FALL 2020 Project-cats
  11. File Systems Unfit as Distributed Storage Backends 开发十年Ceph的经验:文件系统不适合作为分布式存储后端
  12. 误删除恢复 (extundelete)
  13. vue 项目打印时去掉页眉页脚
  14. Linux基础指令入门 2
  15. 微信服务器 系统发生错误,微信系统故障 仍未修复
  16. 淘宝客返利机器人哪个好用点?怎么做?
  17. 计算机网络 IP地址基础知识
  18. 为什么操作系统需要虚拟内存
  19. 四分位数的应用——基于收入实例的箱体图与离群值规则
  20. 伊利诺伊香槟分校计算机排名,伊利诺伊大学香槟分校计算机工程排名第8(2018年TFE美国排名)...

热门文章

  1. 计算机视觉学习6_棋盘格_相机模型与参数标定
  2. Power BI 自定义门户----大成
  3. 手机居然可以变成电脑的无线麦克风?快来瞧瞧
  4. uni-app 表单验证
  5. https://www.cloudmidi.net/ableton-live-11-suite-v11-0-macos.html
  6. 融云钜惠来袭,新客尝鲜首月 2.7 折起,超值套餐 6 折起
  7. Hyperopt的使用
  8. 黑马程序员C++ 第二阶段 核心程序
  9. 个人股权能转让给别人吗
  10. 小技巧:不显示打印对话框直接打印网页到打印机