• Qt 使用RAW INPUT获取HID触摸屏,笔设备,鼠标的原始数据,最低受支持的客户端:Windows XP [仅限桌面应用]


            在开发绘图应用程序时,经常会需要读取笔设备的数据,通过对笔数据的解析,来判断笔的坐标,粗细。如果仅仅只是读取鼠标的坐标,就需要人为在应用程序端去修改笔的粗细,并且使用体验不好,如果可以实时获取到触摸屏的笔数据的话,就可以大大提高体验感。

            获取USB-HID设备的数据,有多种方式

    1. 使用Raw Input API
    2. 使用Windows底层库,如Windows API,Windows HID API
    3. 使用第三方库,如hidapi,libusb等等

            本博文主要分析Raw Input API的操作,官网解析地址:https://learn.microsoft.com/en-us/windows/win32/inputdev/about-raw-input

            从官网的概述可以知道Raw Input API为应用程序提供了一种稳定而可靠的方式,可以接受来自任何 HID(包括键盘和鼠标)的原始输入。

            并且具有这些优点。

    Qt使用Raw Input API具体步骤

    1. 注册原始输入
    2. 重写nativeEvent函数
    3. 执行原始输入的标准读取

    (解析截自官网)

    1.注册原始输入

            为了注册设备,应用程序首先创建 一个 RAWINPUTDEVICE 结构的数组,这些结构为所需的设备指定 顶级集合 (TLC) 。 TLC 由“使用情况页”定义, (设备) 类 (设备) 。 例如,若要获取键盘 TLC,请设置 UsagePage = 0x01 和 UsageID = 0x06。 应用程序调用 RegisterRawInputDevices 来注册设备。

            请注意,应用程序可以注册当前未附加到系统的设备。 附加此设备后,Windows 管理器将自动将原始输入发送到应用程序。 若要获取系统上的原始输入设备列表,应用程序会调用 GetRawInputDeviceList。 应用程序使用此调用中的 hDevice 调用 GetRawInputDeviceInfo 以获取设备信息。

    TCL:

    Usage Page

    HID usages are organized into usage pages of related controls. A specific control usage is defined by its usage page, a usage ID, a name, and a description. A usage page value is a 16-bit unsigned value.

    Examples of usage pages include:

    Expand table

    Page IDPage Namehidusage.h constant
    0x01Generic Desktop ControlsHID_USAGE_PAGE_GENERIC
    0x05Game ControlsHID_USAGE_PAGE_GAME
    0x08LEDsHID_USAGE_PAGE_LED
    0x09ButtonHID_USAGE_PAGE_BUTTON

    Usage ID

    In the context of a usage page, a valid usage identifier, or usage ID, indicates a usage in a usage page. A usage ID of zero is reserved. A usage ID value is an unsigned 16-bit value.

    Examples of controls that are listed on the Generic Desktop Controls usage page:

    Expand table

    Usage IDUsage Namehidusage.h constant
    0x01PointerHID_USAGE_GENERIC_POINTER
    0x02MouseHID_USAGE_GENERIC_MOUSE
    0x04JoystickHID_USAGE_GENERIC_JOYSTICK
    0x05Game PadHID_USAGE_GENERIC_GAMEPAD
    0x06KeyboardHID_USAGE_GENERIC_KEYBOARD
    0x07KeypadHID_USAGE_GENERIC_KEYPAD
    0x08Multi-axis ControllerHID_USAGE_GENERIC_MULTI_AXIS_CONTROLLER

    1. bool Dialog::RegisitWindow(HWND hwnd)
    2. {
    3. if ( nullptr == hwnd ){
    4. return FALSE;
    5. }
    6. RAWINPUTDEVICE rid[1];
    7. rid[0].usUsagePage = 0x0D;
    8. rid[0].usUsage = 0x02; // 笔设备
    9. rid[0].dwFlags = RIDEV_INPUTSINK;
    10. rid[0].hwndTarget = hwnd;
    11. if (!RegisterRawInputDevices(rid, 1, sizeof(rid[0]))){
    12. return FALSE;
    13. }
    14. return TRUE;
    15. }

    2.重写nativeEvent函数

    1. bool Dialog::nativeEvent(const QByteArray &eventType, void *message, long *result)
    2. {
    3. }

    3.执行原始输入的标准读取

            应用程序从任何 HID 接收原始输入,其 顶级集合 (TLC) 与注册中的 TLC 匹配。 当应用程序收到原始输入时,其消息队列将获取 WM_INPUT 消息, 并且队列 状态标志QS_RAWINPUT设置为 (QS_INPUT 还包括此标志) 。 应用程序在前台和后台时可以接收数据。

    有两种方法可以读取原始数据:无缓冲 (或标准) 方法和缓冲方法。 无缓冲区方法一次获取一个 RAWINPUT 结构的原始数据,并且足以用于许多 HID。 在这里,应用程序调用 GetMessage 以获取 WM_INPUT 消息。 然后,应用程序使用包含在 WM_INPUT 中的 RAWINPUT 句柄调用 GetRawInputData。 有关示例,请参阅 执行原始输入的标准读取

    相比之下,缓冲方法一次获取 RAWINPUT 结构的数组。 这是为可以生成大量原始输入的设备提供的。 在此方法中,应用程序调用 GetRawInputBuffer 来获取 RAWINPUT 结构的数组。 请注意, NEXTRAWINPUTBLOCK 宏用于遍历 RAWINPUT 结构的数组。 有关示例,请参阅 执行原始输入的缓冲读取

    若要解释原始输入,需要有关 HID 的详细信息。 应用程序通过使用设备句柄调用 GetRawInputDeviceInfo 来获取设备信息。 此句柄可以来自 WM_INPUT,也可以来自 RAWINPUTHEADER 的 hDevice 成员。

    1. bool Dialog::nativeEvent(const QByteArray &eventType, void *message, long *result)
    2. {
    3. MSG* msg = (MSG*)message;
    4. LPBYTE lpb;
    5. RAWINPUT* raw;
    6. UINT dwSize;
    7. if(msg->message == WM_INPUT){
    8. GetRawInputData((HRAWINPUT)msg->lParam, RID_INPUT, nullptr, &dwSize, sizeof(RAWINPUTHEADER));
    9. lpb = new BYTE[dwSize];
    10. if (lpb == nullptr) {
    11. qDebug() << "内存分配失败。";
    12. return false;
    13. }
    14. if (GetRawInputData((HRAWINPUT)msg->lParam, RID_INPUT, lpb, &dwSize, sizeof(RAWINPUTHEADER)) != dwSize) {
    15. qDebug() << "GetRawInputData 返回的大小不正确!";
    16. delete[] lpb;
    17. return false;
    18. }
    19. raw = (RAWINPUT*)lpb;
    20. if (raw->header.dwType == RIM_TYPEHID) {
    21. qDebug() << "接收到 HID 输入。";
    22. //在这里可以使用raw->data.hid.bRawData解析数据。
    23. }
    24. delete[] lpb;
    25. }
    26. return false;
    27. }

  • 相关阅读:
    elasticsearch 自允许动创建索引
    开源游戏服务器框架NoahGameFrame(NF)服务器端环境搭建(二)
    Spring IOC核心功能快速入门
    [ZJOI2009]假期的宿舍
    openGauss学习笔记-100 openGauss 数据库管理-管理数据库安全-客户端接入之用SSL进行安全的TCP/IP连接
    Spring MVC处理用户请求的完整流程
    一文搞懂Java中的继承(父类与子类间的联系)+实例分析
    Linux修改24小时制中国时区
    MATLAB算法实战应用案例精讲-【图像处理】机器视觉(应用篇)
    mysql练习1
  • 原文地址:https://blog.csdn.net/weixin_62370184/article/details/136650409