12

我写了这段代码来观察按键动作的事件。问题似乎是当这个脚本运行时,某些程序会崩溃这个程序,吐出这个错误信息:

TypeError: KeyboardSwitch() missing 8 required positional arguments: 'msg', 'vk_
code', 'scan_code', 'ascii', 'flags', 'time', 'hwnd', and 'win_name'

观察到崩溃的一些程序是:Skype、Sublime Text 2

经过几次调试试验后,问题似乎出现在最后一行,但我似乎无法缩小范围。我也不明白编译器返回的 KeyboardSwitch() 的含义......

我还发现该程序会交替返回此错误消息

Traceback (most recent call last):
  File "C:\Python34\lib\site-packages\pyHook\HookManager.py", line 351, in KeyboardSwitch
    return func(event)
  File "observe.py", line 6, in OnKeyboardEvent
    print ('MessageName:',event.MessageName)
TypeError: an integer is required (got type NoneType)

是什么原因以及如何解决这个问题,特别是因为它只出现在 2 个按键中的 1 个被按下的情况下

import pyHook, pythoncom

def OnKeyboardEvent(event):
# Source: http://code.activestate.com/recipes/553270-using-pyhook-to-block-windows-keys/ 
    print ('MessageName:',event.MessageName)
    print ('Message:',event.Message)
    print ('Time:',event.Time)
    print ('Window:',event.Window)
    print ('WindowName:',event.WindowName)
    print ('Ascii:', event.Ascii, chr(event.Ascii))
    print ('Key:', event.Key)
    print ('KeyID:', event.KeyID)
    print ('ScanCode:', event.ScanCode)
    print ('Extended:', event.Extended)
    print ('Injected:', event.Injected)
    print ('Alt', event.Alt)
    print ('Transition', event.Transition)
    print ('---')

hooks_manager = pyHook.HookManager()
hooks_manager.KeyDown = OnKeyboardEvent
hooks_manager.HookKeyboard()
pythoncom.PumpMessages()

PS 作为一个初学者,我对pythoncom的功能不是很熟悉,网上的定义也比较模糊。非常感谢对 pythoncom 和 PumpMessages 功能的解释。

谢谢

4

3 回答 3

9

我认为问题在于,当 pyHook 被 Windows 回调时,它所做的第一件事就是获取具有焦点的窗口的窗口名称。

PSTR win_name = NULL;
...
// grab the window name if possible
win_len = GetWindowTextLength(hwnd);
if(win_len > 0) {
  win_name = (PSTR) malloc(sizeof(char) * win_len + 1);
  GetWindowText(hwnd, win_name, win_len + 1);
}

所以我认为这里的问题是,即使GetWindowText不返回宽字符,它也可以从 ANSI 代码页返回非 ascii 字符。然而,在我们这样做之前,这不会失败:

// pass the message on to the Python function
arglist = Py_BuildValue("(iiiiiiiz)", wParam, kbd->vkCode, kbd->scanCode, ascii,
                        kbd->flags, kbd->time, hwnd, win_name);

在这里,由于z格式字符串中的 ,win_name变量中的数据被转换为 unicodestrPy_BuildValue假设它是 ASCII。但它不是:所以它可以触发一个UnicodeDecodeError. 然后,这会导致arglist成为NULL,因此您的函数被调用而没有参数。

所以我不完全确定这里的最佳解决方案。但是我只是将这两个代码都更改为使用宽字符和 unicode 而不是 ascii,并重新构建了 pyHook,这似乎解决了它。我认为它只能在 Python 3 版本中使用,但对于 Python 2,我认为旧的 pyHook 仍然可以使用。

LPWSTR win_name = NULL;

...
// grab the window name if possible
win_len = GetWindowTextLengthW(hwnd);
if(win_len > 0) {
  win_name = (LPWSTR) malloc(sizeof(wchar_t) * win_len + 1);
  GetWindowTextW(hwnd, win_name, win_len + 1);
}

// pass the message on to the Python function
arglist = Py_BuildValue("(iiiiiiiu)", wParam, kbd->vkCode, kbd->scanCode, ascii,
                        kbd->flags, kbd->time, hwnd, win_name);

该问题仅出现在标题中包含非 ascii 字符的窗口中:Skype 就是其中之一。

于 2015-10-26T23:10:40.450 回答
0

如果每 2 次印刷机中只有 1 次有效,则绝对是缺少返回值的问题。尝试返回 True 或 False。

于 2014-12-05T11:28:41.210 回答
-1

python3有一个pyhook:https ://github.com/Answeror/pyhook_py3k 这个bug已经在这个项目中得到修复。

于 2017-11-03T03:05:25.237 回答