在Python中简化Ctype联合(在Windows中发送键盘事件)

2024-09-29 19:19:58 发布

您现在位置:Python中文网/ 问答频道 /正文

我有下面的脚本,它每xx秒发送一个键盘键(默认情况下是F15),灵感来自于在线找到的一些代码

我想删除与鼠标事件相关的类型和联合,但无法使其工作。尤其是,我不知道如何删除类MOUSEINPUT和类型的联合_INPUTunion(删除它们将停止发送键盘键)。在

关于如何将脚本缩减到最小值(即只保留与键盘相关的代码)的建议吗?在

下面的代码将发送键“C”,以便能够进行调试。在

#!/python

import ctypes
import sys
import time

LONG = ctypes.c_long
DWORD = ctypes.c_ulong
ULONG_PTR = ctypes.POINTER(DWORD)
WORD = ctypes.c_ushort

class MOUSEINPUT(ctypes.Structure):
    _fields_ = (
        ('dx', LONG), ('dy', LONG), ('mouseData', DWORD),
        ('dwFlags', DWORD), ('time', DWORD),
        ('dwExtraInfo', ULONG_PTR)
    )

class KEYBDINPUT(ctypes.Structure):
    _fields_ = (
        ('wVk', WORD), ('wScan', WORD),
        ('dwFlags', DWORD), ('time', DWORD),
        ('dwExtraInfo', ULONG_PTR)
    )

class _INPUTunion(ctypes.Union):
    _fields_ = (('mi', MOUSEINPUT), ('ki', KEYBDINPUT))

class INPUT(ctypes.Structure):
    _fields_ = (('type', DWORD), ('union', _INPUTunion))


def SendInput(*inputs):
    print(inputs[0].union.mi)
    nInputs = len(inputs)
    LPINPUT = INPUT * nInputs
    pInputs = LPINPUT(*inputs)
    cbSize = ctypes.c_int(ctypes.sizeof(INPUT))
    return ctypes.windll.user32.SendInput(nInputs, pInputs, cbSize)

INPUT_keyboard = 1

def Input(structure):
    if isinstance(structure, KEYBDINPUT):
        return INPUT(INPUT_KEYBOARD, _INPUTunion(ki=structure))
    else:
        raise TypeError('Cannot create INPUT structure (keyboard)!')

def Keyboard(code, flags=0):
    return Input(KEYBDINPUT(code, code, flags, 0, None))

if __name__ == '__main__':
    nb_cycles = 10
    while nb_cycles != 0:
            time.sleep(2)  # 3 seconds
            # Key "c" for debug, but ideally use 0x7E for "F15"
            SendInput(Keyboard(ord("C")))
            sys.stdout.write(".")
            nb_cycles -= 1

Tags: 代码importfieldsinputtime键盘ctypesstructure
1条回答
网友
1楼 · 发布于 2024-09-29 19:19:58

像这样任务的“圣经”:[Python 3]: ctypes - A foreign function library for Python
我修改了你的代码(发现了一堆问题,其中一些是关键问题)。在

代码.py

#!/usr/bin/env python3

import sys
import time
import ctypes
from ctypes import wintypes


class KEYBDINPUT(ctypes.Structure):
    _fields_ = [
        ("wVk", wintypes.WORD),
        ("wScan", wintypes.WORD),
        ("dwFlags", wintypes.DWORD),
        ("time", wintypes.DWORD),
        ("dwExtraInfo", ctypes.POINTER(wintypes.ULONG)),
    ]


class INPUT(ctypes.Structure):
    _fields_ = [
        ("type", wintypes.DWORD),
        ("ki", KEYBDINPUT),
        ("padding", ctypes.c_ubyte * 8)
    ]


INPUT_KEYBOARD = 1  # Also defined by win32con if you have pywin32 installed

INPUT_LEN = ctypes.sizeof(INPUT)
LPINPUT = ctypes.POINTER(INPUT)

SendInput = ctypes.windll.user32.SendInput
SendInput.argtypes = [wintypes.UINT, LPINPUT, ctypes.c_int]
SendInput.restype = wintypes.UINT


def send_input(_input):
    return SendInput(1, ctypes.byref(_input), INPUT_LEN)


def keyboard(code, flags=0):
    return INPUT(INPUT_KEYBOARD, (KEYBDINPUT(code, 0, flags, 0, None)))


def main():
    time.sleep(2)
    nb_cycles = 3
    for _ in range(nb_cycles):
        time.sleep(0.5)  # 3 seconds
        # Key "c" for debug, but ideally use 0x7E for "F15"
        ret = send_input(keyboard(ord("C")))
        #print(ret)
        sys.stdout.write(".")
        sys.stdout.flush()


if __name__ == "__main__":
    print("Python {:s} on {:s}\n".format(sys.version, sys.platform))
    main()

注意事项

  • 最初,我定义了只有1st2个成员的INPUT结构,但是查看了[MS.Docs]: INPUT structure,并注意到该联合包括:

    • 鼠标输入
    • KEYBDINPUT
    • 硬件输入

    做了一些测试并注意到MOUSEINPUT的大小在3struct中是最大的,而且它的8字节大于KEYBDINPUT(对于两个32位64位),所以我添加了(dummy)(padding)成员。通常,当接收到一个type设置为INPUT峈KEYBOARDINPUT结构时,我不会期望SendInput超过KEYBDINPUT大小,但是INPUT大小是[MS.Docs]: SendInput functioncbSizearg

  • def SendInput(*inputs):-在Python中,*之前,参数所做的事情与C中完全不同。检查[SO]: Expanding tuples into arguments(我没有找到官方文件)。我修改了函数,使它只发送一个这样的结构

  • 始终为通过ctypes调用的函数定义argtypes(和restype)。否则它们将默认为ctypes.c_int(这可能会导致一些严重的错误,特别是在64位[SO]: Python ctypes cdll.LoadLibrary, instantiate an object, execute its method, private variable address truncated (@CristiFati's answer)

  • ctypes.wintypes用于Win的特定类型,不要重新设计轮子。更不用说有时会出现不匹配(我最近看到的一个例子是ctypes.c_boolwintypes.BOOL

  • 我将您的函数重命名为[Python]: PEP 8 Style Guide for Python Code兼容。我还删除了一些不再需要的文件

  • 其他不值得单独提及的小变化

至于测试,我使用了一个记事本窗口(或任何其他对用户输入“敏感”的窗口):

  • 启动脚本
  • Alt+Tab到测试窗口(我添加了main中的1st指令(time.sleep(2)),只是为了给用户时间切换窗口),然后注意到cs“神奇地”出现了

。。。或者只需从控制台启动脚本,按Ctrl,注意它被取消了。在

相关问题 更多 >

    热门问题