在之前写的一篇[《Python:监控键盘输入、鼠标操作,并将捕获到的信息记录到文件中》](http://blog.csdn.net/dyx1024/article/details/7311013)文章中,有个读者留言如下:
![](https://box.kancloud.cn/2016-06-08_575793590ab34.gif)
这看似一个很平常的需求,但实现起来并不容易,如果用快捷键来控制一个程序干些别的事情那是非常容易的,但关键是本程序刚好是用hook来监控键盘,所以必须使用PumpMessages(),而此函数使用当前程序进入消息循环,它抓取每个鼠标和键盘事件。当我们的程序跑起来后,按下停止的热键时,也被此函数捕获,所以定义的任何热键均不能生效,具体实现及测试在文章[《Python:通过自定义系统级快捷键来控制程序运行》](http://blog.csdn.net/dyx1024/article/details/7335085)中有所描述。
现在,我们换一个思路,既然已经监控到了按键,那就判断当前的按键是不是预先定义的热键,如果是,则可调用自己的处理函数,这样就找到了一个控制的入口,可以通过它实现我们想要的功能,注意此时不能让程序调用os.exit(0)让程序退出,否则再次按启动热键时就没法玩了。具体实现如下,代码中有详细注释,不再一一解释。
### 一、代码:
~~~
#!/usr/bin/env python
# -*- coding: utf-8 -*-
import pythoncom
import pyHook
import time
import pyhk
import os
import sys
import ctypes
from ctypes import wintypes
import win32con
import win32api
class CInspectKeyAndMouseEvent:
'''
Function:键盘和鼠标监控类
Input:NONE
Output: NONE
author: socrates
blog:http://blog.csdn.net/dyx1024
date:2012-03-09
'''
def __init__(self, filename):
'初始化'
self.filename = filename
def open_file(self):
'打开文件'
self.fobj = open(self.filename, 'w')
def close_file(self):
'关闭文件'
self.fobj.close()
def IsNotWriteLog(self):
'是否记录日志'
return self.bFlag
def IsExitCommand(self, event):
'''
是否当前按下了程序定义的热键'
如果按下了ALT+F2,将记录日志的状态位置为True,不记录日志,
如果按下了ALT+F1,将记录日志状态位置为False,表示记录日志
'''
if event.Alt == 32 and str(event.Key) == 'F2':
self.bFlag = True
print time.strftime('[%Y-%m-%d %H:%M:%S]: ',time.localtime(time.time()))+ ' stop write log'
elif event.Alt == 32 and str(event.Key) == 'F1':
self.bFlag = False
print time.strftime('[%Y-%m-%d %H:%M:%S]: ',time.localtime(time.time()))+ ' start write log'
def onMouseEvent(self, event):
"处理鼠标事件"
#判断是否要记录日志
if self.IsNotWriteLog():
return True
self.fobj.writelines('-' * 20 + 'MouseEvent Begin' + '-' * 20 + '\n')
self.fobj.writelines("Current Time:%s\n" % time.strftime('[%Y-%m-%d %H:%M:%S]: ',time.localtime(time.time())))
self.fobj.writelines("MessageName:%s\n" % str(event.MessageName))
self.fobj.writelines("Message:%d\n" % event.Message)
self.fobj.writelines("Time_sec:%d\n" % event.Time)
self.fobj.writelines("Window:%s\n" % str(event.Window))
self.fobj.writelines("WindowName:%s\n" % str(event.WindowName))
self.fobj.writelines("Position:%s\n" % str(event.Position))
self.fobj.writelines('-' * 20 + 'MouseEvent End' + '-' * 20 + '\n')
return True
def onKeyboardEvent(self, event):
#处理按下的热键
self.IsExitCommand(event)
#判断是否要记录日志
if self.IsNotWriteLog():
return True
self.fobj.writelines('-' * 20 + 'Keyboard Begin' + '-' * 20 + '\n')
self.fobj.writelines("Current Time:%s\n" % time.strftime('[%Y-%m-%d %H:%M:%S]: ',time.localtime(time.time())))
self.fobj.writelines("MessageName:%s\n" % str(event.MessageName))
self.fobj.writelines("Message:%d\n" % event.Message)
self.fobj.writelines("Time:%d\n" % event.Time)
self.fobj.writelines("Window:%s\n" % str(event.Window))
self.fobj.writelines("WindowName:%s\n" % str(event.WindowName))
self.fobj.writelines("Ascii_code: %d\n" % event.Ascii)
self.fobj.writelines("Ascii_char:%s\n" % chr(event.Ascii))
self.fobj.writelines("Key:%s\n" % str(event.Key))
self.fobj.writelines('-' * 20 + 'Keyboard End' + '-' * 20 + '\n')
return True
#默认记录
bFlag = False
def InspectKeyAndMouseEvent():
"启动监控"
my_event = CInspectKeyAndMouseEvent("D:\\hook_log.txt")
my_event.open_file()
#创建hook句柄
hm = pyHook.HookManager()
#监控键盘
hm.KeyDown = my_event.onKeyboardEvent
hm.HookKeyboard()
#监控鼠标
hm.MouseAll = my_event.onMouseEvent
hm.HookMouse()
#循环获取消息
pythoncom.PumpMessages()
my_event.close_file()
def handle_start_InspecEvent():
"开始监控(按下Ctrl + F1)"
print time.strftime('[%Y-%m-%d %H:%M:%S]: ',time.localtime(time.time()))+ ' start write log'
InspectKeyAndMouseEvent()
#def handle_stop_InspecEvent():
# "停止监控 (按下Ctrl + F2)"
# InspectKeyAndMouseEvent(False)
if __name__ == "__main__":
'''
Function:通过快捷键控制程序运行
Input:NONE
Output: NONE
author: socrates
blog:http://blog.csdn.net/dyx1024
date:2012-03-09
'''
byref = ctypes.byref
user32 = ctypes.windll.user32
#定义快捷键
HOTKEYS = {
1 : (win32con.VK_F1, win32con.MOD_ALT)
# 2 : (win32con.VK_F2, win32con.MOD_ALT)
}
#快捷键对应的驱动函数
HOTKEY_ACTIONS = {
1 : handle_start_InspecEvent,
# 2 : handle_stop_InspecEvent
}
#注册快捷键
for id, (vk, modifiers) in HOTKEYS.items ():
if not user32.RegisterHotKey (None, id, modifiers, vk):
print "Unable to register id", id
#启动监听
try:
msg = wintypes.MSG ()
while user32.GetMessageA (byref (msg), None, 0, 0) != 0:
if msg.message == win32con.WM_HOTKEY:
action_to_take = HOTKEY_ACTIONS.get (msg.wParam)
if action_to_take:
action_to_take ()
user32.TranslateMessage (byref (msg))
user32.DispatchMessageA (byref (msg))
finally:
for id in HOTKEYS.keys ():
user32.UnregisterHotKey (None, id)
~~~
### 二、测试:
1、以下打印是按下热键时控制台输出(支持当前程序不是非激活窗口下按下热键)
![](https://box.kancloud.cn/2016-06-08_575793592ba94.gif)
2、日志内容:
可以看到,在23:16:58停止记录日志后,至23:17:05重新开始记录之前,所有的键盘和鼠标输入均没有记录,达到预期效果。
~~~
--------------------MouseEvent Begin--------------------
Current Time:[2012-03-09 23:16:57]:
MessageName:mouse move
Message:512
Time_sec:12542031
Window:328916
WindowName:FolderView
Position:(737, 438)
--------------------MouseEvent End--------------------
--------------------Keyboard Begin--------------------
Current Time:[2012-03-09 23:16:58]:
MessageName:key sys down
Message:260
Time:12542890
Window:1639322
WindowName:本地磁盘 (D:)
Ascii_code: 0
Ascii_char:
Key:Lmenu
--------------------Keyboard End--------------------
--------------------Keyboard Begin--------------------
Current Time:[2012-03-09 23:17:05]:
MessageName:key sys down
Message:260
Time:12550015
Window:1639322
WindowName:本地磁盘 (D:)
Ascii_code: 0
Ascii_char:
Key:F1
--------------------Keyboard End--------------------
--------------------MouseEvent Begin--------------------
Current Time:[2012-03-09 23:17:06]:
MessageName:mouse move
Message:512
Time_sec:12551000
Window:328916
WindowName:FolderView
Position:(720, 420)
--------------------MouseEvent End--------------------
--------------------MouseEvent Begin--------------------
Current Time:[2012-03-09 23:17:06]:
MessageName:mouse move
Message:512
Time_sec:12551015
Window:328916
WindowName:FolderView
Position:(719, 420)
--------------------MouseEvent End--------------------
~~~
- 前言
- Python:实现文件归档
- Pyhon:按行输出文件内容
- Python:读文件和写文件
- Python:实现一个小算法
- Python:通过命令行发送新浪微博
- Python:通过摄像头实现的监控功能
- Python:通过摄像头抓取图像并自动上传至新浪微博
- Python:简单的摄像头程序实现
- Python:日志模块logging的应用
- Python:操作嵌入式数据库SQLite
- Python:将句子中的单词全部倒排过来,但单词的字母顺序不变
- Python:语音处理,实现在线朗读RFC文档或本地文本文件
- Python:通过计算阶乘来学习lambda和reduce这两个函数的使用
- Python:通过执行100万次打印来比较C和python的性能,以及用C和python结合来解决性能问题的方法
- Python:使用matplotlib绘制图表
- Python:使用pycha快速绘制办公常用图(饼图、垂直直方图、水平直方图、散点图等七种图形)
- Python:使用pycha快速绘制办公常用图二(使用样式定制个性化图表)
- Python:监控键盘输入、鼠标操作,并将捕获到的信息记录到文件中
- Python:通过获取淘宝账号和密码的实验,来看登陆方式选择的重要性
- Python:通过获取淘宝账号和密码的实验,来看登陆方式选择的重要性(二)
- Python:通过远程监控用户输入来获取淘宝账号和密码的实验(一)
- Python:通过远程监控用户输入来获取淘宝账号和密码的实验(二)
- Python:通过自定义系统级快捷键来控制程序运行
- Python:通过自定义系统级快捷键来控制程序开始或停止记录日志(使用小技巧解决一个貌似无解的问题)
- Python:一个多功能的抓图工具开发(附源码)
- Python:程序发布方式简介一(打包为可执行文件EXE)
- Python:新浪微博应用开发简介(认证及授权部分)
- Python:程序最小化到托盘功能实现
- Python:实用抓图工具开发介绍(含需求分析、设计、编码、单元测试、打包、系统测试、发布各环节)
- Python:桌面气泡提示功能实现
- Python:未来三个月的python学习计划
- Python:pygame模块及SDL库简介
- Python:获取新浪微博用户的收听列表和粉丝列表
- Python:pygame游戏编程之旅一(Hello World)
- Python:pygame游戏编程之旅二(自由移动的小球)
- Python:pygame游戏编程之旅三(玩家控制的小球)
- Python:pygame游戏编程之旅四(游戏界面文字处理)
- Python:pygame游戏编程之旅五(游戏界面文字处理详解)
- Python:pygame游戏编程之旅六(游戏中的声音处理)
- Python:pygame游戏编程之旅七(pygame基础知识讲解1)
- Python:编程“八荣八耻”之我见
- Python:脚本的几种执行方式
- wxPython:简单的wxPython程序
- wxPython:简单的wxPython程序的另一种写法
- wxPython:应用程序对象介绍
- wxPython:输出重定向
- wxPython:关闭wxPython程序
- wxPython:Frame类介绍
- wxPython:面板Panel的使用
- wxPython:工具栏、状态栏、菜单实现
- wxPython:消息对话框MessageDialog
- wxPython:文本对话框TextEntryDialog
- wxPython:列表选择框SingleChoiceDialog
- wxPython:事件处理介绍一
- wxPython:事件处理介绍二
- wxPython: 简单的绘图例子
- wxPython:状态栏介绍
- wxPython:菜单介绍
- wxPython:文件对话框wx.FileDialog
- wxPython:颜色选择对话框wx.ColourDialog
- wxPython:布局管理器sizer介绍
- wxPython:启动画面SplashScreen介绍
- wxPython:绘画按钮BitmapButton介绍
- wxPython:进度条Gauge介绍
- Python: 发送新浪微博(使用oauth2)
- Python:读取新浪微博收听列表
- Python:DNS客户端实现