用python制作一款录屏小工具

天我想学习记录的内容是 ——如何用python实现录屏

1

写在前面

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

2

效果展示

3

知识串讲(敲黑板啦)

这次要讲的东西可能比较多了,涉及到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帧画面会不停地显示在你桌面上!

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

4

示例代码

1、工具GUI界面代码:

# coding=utf-8# @Auther : "鹏哥贼优秀"# @Date : 2019/11/27# @Software : PyCharm import sysfrom PyQt5.QtCore import *from PyQt5.QtWidgets import *import timeimport win32api,win32confrom 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):        # 垂直布局类QVBoxLayout        layout = QVBoxLayout(top)        # 添加录屏相关按钮        button1 = QPushButton("自定义录屏")        layout.addWidget(button1)        button2 = QPushButton("全屏录屏")        layout.addWidget(button2)        button3 = QPushButton("停止录屏")        layout.addWidget(button3)        self.text = QPlainTextEdit('欢迎使用!By 鹏哥贼优秀')        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_())

2、录屏函数

# coding=utf-8# @Auther : "鹏哥贼优秀"# @Date : 2019/11/27# @Software : PyCharmfrom PIL import ImageGrabimport numpy as npimport cv2import osimport jpypedef Recording(tag=1):    # 录屏开始时创建test.txt,作为结束录屏的条件    if 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 = None    image = ImageGrab.grab(record_region)  # 获取指定范围的屏幕对象    width, height = image.size    fourcc = 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 regiondef StopRecording():    os.remove('test.txt') #停止录屏的触发条件if __name__ == "__main__":    Recording()

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,636评论 5 468
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,890评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,680评论 0 330
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,766评论 1 271
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,665评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,045评论 1 276
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,515评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,182评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,334评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,274评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,319评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,002评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,599评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,675评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,917评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,309评论 2 345
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,885评论 2 341

推荐阅读更多精彩内容