首页 > 解决方案 > 如何防止 pynput 和 ctypes 发生冲突?

问题描述

我在这个网站的某个地方使用这个 gem。

import ctypes
import pynput


SendInput = ctypes.windll.user32.SendInput

W = 0x11
A = 0x1E
S = 0x1F
D = 0x20

# C struct redefinitions 
PUL = ctypes.POINTER(ctypes.c_ulong)
class KeyBdInput(ctypes.Structure):
    _fields_ = [("wVk", ctypes.c_ushort),
                ("wScan", ctypes.c_ushort),
                ("dwFlags", ctypes.c_ulong),
                ("time", ctypes.c_ulong),
                ("dwExtraInfo", PUL)]

class HardwareInput(ctypes.Structure):
    _fields_ = [("uMsg", ctypes.c_ulong),
                ("wParamL", ctypes.c_short),
                ("wParamH", ctypes.c_ushort)]

class MouseInput(ctypes.Structure):
    _fields_ = [("dx", ctypes.c_long),
                ("dy", ctypes.c_long),
                ("mouseData", ctypes.c_ulong),
                ("dwFlags", ctypes.c_ulong),
                ("time",ctypes.c_ulong),
                ("dwExtraInfo", PUL)]

class Input_I(ctypes.Union):
    _fields_ = [("ki", KeyBdInput),
                 ("mi", MouseInput),
                 ("hi", HardwareInput)]

class Input(ctypes.Structure):
    _fields_ = [("type", ctypes.c_ulong),
                ("ii", Input_I)]

# Actuals Functions

def PressKey(hexKeyCode):
    extra = ctypes.c_ulong(0)
    ii_ = Input_I()
    ii_.ki = KeyBdInput( 0, hexKeyCode, 0x0008, 0, ctypes.pointer(extra) )
    x = Input( ctypes.c_ulong(1), ii_ )
    ctypes.windll.user32.SendInput(1, ctypes.pointer(x), ctypes.sizeof(x))

def ReleaseKey(hexKeyCode):
    extra = ctypes.c_ulong(0)
    ii_ = Input_I()
    ii_.ki = KeyBdInput( 0, hexKeyCode, 0x0008 | 0x0002, 0, ctypes.pointer(extra) )
    x = Input( ctypes.c_ulong(1), ii_ )
    ctypes.windll.user32.SendInput(1, ctypes.pointer(x), ctypes.sizeof(x))


# directx scan codes http://www.gamespp.com/directx/directInputKeyboardScanCodes.html
# ganna need to rework pynput for this to work
import time


def asdf():
    while True:
        PressKey(0x11)
        time.sleep(1)
        ReleaseKey(0x11)
        time.sleep(1)

asdf()

但是只要导入 pynput,就会返回这个错误。

ctypes.ArgumentError:参数 2::预期 LP_INPUT 实例而不是 LP_Input

这个小 ctypes 脚本确实可以独立工作,但我真的想尝试将这些机制合并到我的程序的其余部分中。我不想废弃我的代码的 pynput 部分。它变得相当大。

有什么方法可以阻止他们尝试相互合作吗?因为我认为这是因为 pynput 更像是一个包装器,并且在某种程度上增加了它提取的数据。具体不太清楚,还在学习中。

我需要 ctypes 的原因是因为它是我发现的唯一输出直接输入的解决方案。(适用于游戏和任何使用 directx 的东西。)对不起,如果这没有足够的信息或者我以丑陋的方式发布了这个。我愿意通过建议来解决这个问题。

更新:

去学习 C.

这是其余的错误。

回溯(最近一次通话):文件“C:/Users/bbdan/PycharmProjects/Playground/directkeys.py”,第 72 行,在 asdf() 文件“C:/Users/bbdan/PycharmProjects/Playground/directkeys.py”中, 第 67 行, asdf PressKey(0x11) 文件 "C:/Users/bbdan/PycharmProjects/Playground/directkeys.py", 第 50 行, 在 PressKey ctypes.windll.user32.SendInput(1, ctypes.pointer(x), ctypes.sizeof(x)) ctypes.ArgumentError: 参数 2: : 预期的 LP_INPUT 实例而不是 LP_Input

标签: python-3.xdirectxctypesdirectinputpynput

解决方案


pip install input和它玩了一下。我的猜测是正确的,Pynput定义了这些结构,但名称略有不同,并将argtypes(和restype)设置ctypes.windll.user32.SendInput为它自己的定义。
这就是为什么当您尝试提供结构的实例时,它会看到它抱怨类型不匹配。

有许多解决方案可以解决此问题。无论如何,最简单的一种是简单地用Pynput替换你的结构(你不再需要它们)。

注意:这只是一个愚蠢的替换,事情可以组织得更好,而且我确信Pynput有自己的机制来实现这一点,以便让用户不必编写这段代码。

PressKeyReleaseKey的 2 个修改版本:

def PressKeyPynput(hexKeyCode):
    extra = ctypes.c_ulong(0)
    ii_ = pynput._util.win32.INPUT_union()
    ii_.ki = pynput._util.win32.KEYBDINPUT(0, hexKeyCode, 0x0008, 0, ctypes.cast(ctypes.pointer(extra), ctypes.c_void_p))
    x = pynput._util.win32.INPUT(ctypes.c_ulong(1), ii_)
    SendInput(1, ctypes.pointer(x), ctypes.sizeof(x))

def ReleaseKeyPynput(hexKeyCode):
    extra = ctypes.c_ulong(0)
    ii_ = pynput._util.win32.INPUT_union()
    ii_.ki = pynput._util.win32.KEYBDINPUT(0, hexKeyCode, 0x0008 | 0x0002, 0, ctypes.cast(ctypes.pointer(extra), ctypes.c_void_p))
    x = pynput._util.win32.INPUT(ctypes.c_ulong(1), ii_)
    SendInput(1, ctypes.pointer(x), ctypes.sizeof(x))

注意:此处遇到的(更复杂的)变体:[SO]: ctypes.ArgumentError when using kivy with pywinauto


推荐阅读