毕设有一部分要用到类似QQ截图的功能,这里记录制作过程。因为后期要添加人工智能的功能,所以用python来写桌面应用。
可以简单的将过程分为以下三步:
现在我将从第三步开始一步步向上搜索、编写、验证。
通过对“python”和“截图”两个关键字搜索不难发现python有多种满足我们需求的函数,这里选用PIL库的grab()函数。time函数是用来测试截图速度的。
from PIL import ImageGrab
import time
# 截取图像
def catch_area(left, top, right, bottom):
return ImageGrab.grab((left, top, right, bottom))
# 测试函数
def test():
first = time.time()
catch_area(101,101,200,300).save('D:/1.jpg')
print(time.time() - first)
test()
截图函数需要四个参数,即矩形区域的上下左右的坐标。为了获取这四个参数我们需要监听鼠标的点击事件。我本来想搜寻一个全局监听器的api的,这里选用的是pyhook。但是测试发现不支持python2.7,而python3.7的安装过程极其复杂,故放弃此方法。
后来我想到可以利用GUI生成一个全屏的button,然后监听对这个button的点击事件。这里的GUI选用wxPython,3.7版本直接pip安装。
import wx
from PIL import ImageGrab
class ScreenShot(wx.Frame):
left,right,top,bottom = 0,0,0,0
img = None
def __init__(self, parent):
wx.Frame.__init__(self, parent,
style = wx.MAXIMIZE # 全屏显示
)
# 设置背景色
self.SetBackgroundColour((255,255,255))
# 设置透明度
self.SetTransparent(30)
# 注册事件
self.Bind(wx.EVT_LEFT_DOWN, self.OnDown, self)
self.Bind(wx.EVT_LEFT_UP, self.OnUp, self)
# 显示button
self.Show(True)
def OnDown(self, event):
pos = event.GetPosition()
self.top = pos.y
self.left = pos.x
def OnUp(self, event):
pos = event.GetPosition()
self.bottom = pos.y
self.right = pos.x
print(self.left, self.top, self.right, self.bottom)
self.img = self.catch_area(
self.left, self.top, self.right, self.bottom)
# 截图完毕后关闭button
self.Close(False)
def catch_area(self, left, top, right, bottom):
return ImageGrab.grab((left, top, right, bottom))
# 测试用例
# app = wx.App(False)
# frame = ScreenShot(None)
# app.MainLoop()
这里只是将截得图像存在img变量中,至于后续操作请各位自行实现。
最后需要构造一个按钮并为其设置快捷键/热键。GUI同样使用之前的wxPython。页面就随便写写;设置快捷键即为某个控件绑定键盘事件,需要将焦点聚集到某个控件上;而热键是可以全局使用,热键设置可将事件绑定到菜单选项上,也可以直接注册。这里直接注册热键。
import wx
import ScreenShot as SS
class MainWindow(wx.Frame):
def __init__(self, parent):
# 视窗属性
wx.Frame.__init__(self, parent,
size = (1000, 200), # 视窗大小
style = wx.CLIP_CHILDREN, # 视窗样式,这个样式可以去掉原生的标题栏
#pos = (200, 200) # 视窗位置,下面设置居中了
)
# 创建位于窗口的底部的状态栏
self.CreateStatusBar()
# 布局
self.root = wx.BoxSizer(wx.VERTICAL)
self.menu = wx.BoxSizer(wx.HORIZONTAL)
self.body = wx.BoxSizer(wx.VERTICAL)
self.menuText = [u"开始截图", u"退出"]
self.menubuttons = []
for i in range(0, 2):
# 新建按键
self.menubuttons.append(wx.Button(self, -1, self.menuText[i]))
self.menu.Add(self.menubuttons[i], 1, wx.SHAPED)
# 新建只读文本框(预留)
self.control = wx.TextCtrl(self, style = wx.TE_READONLY)
self.body.Add(self.control, 1, wx.GROW)
self.root.Add(self.menu, 1, wx.GROW)
self.root.Add(self.body, 4, wx.EXPAND)
# 激活布局
self.SetSizer(self.root)
self.SetAutoLayout(True)
# self.root.Fit(self) # 适应内部大小
# 事件注册
self.Bind(wx.EVT_BUTTON, self.OnStart, self.menubuttons[0])
self.Bind(wx.EVT_BUTTON, self.OnExit, self.menubuttons[1])
# 热键注册
self.hotkey = wx.NewIdRef() # 创建id
self.RegisterHotKey(self.hotkey, wx.MOD_CONTROL, ord('Q')) # 注册热键(按ctrl+Q键响应)
self.Bind(wx.EVT_HOTKEY, self.OnKeyBoard, id=self.hotkey) # 绑定热键事件
# 窗口居中
self.Center()
# 显示视窗
self.Show(True)
# 开始截图
def OnStart(self, event):
SS.ScreenShot(self)
# 关闭整个应用
def OnExit(self, event):
self.Destroy()
# 热键事件
def OnKeyBoard(self, event):
SS.ScreenShot(self)
app = wx.App(False)
frame = MainWindow(None)
app.MainLoop()
最后两个代码块就是完整应用。这只是一个十分粗略的应用,很多细节都没有优化,比如最小化到托盘、选取区域时自动标出区域。若各位有能力可自行实现。
顺带吐槽一下,python的国内学习资料真是太少了,我想搜索一个“wxPython如何设置背景颜色”搜了半天也没搜到什么。又或者百度变拉了?