当前位置: 首页 > ai >正文

C#解析USB - HID手柄上摇杆按键数据

1. 了解相关知识

  • ​HID设备通信原理​​:HID设备通过端点报告其状态和数据,设备通过报告描述符来描述数据用途,操作系统通过这个描述符了解设备发送数据的结构。通常一个完整的报告以特定的格式从设备传输至主机。
  • ​Windows API函数​​:在C#中,可使用Windows API函数通过P/Invoke(平台调用)方式来与HID设备交互。

2. 准备工作

  • ​获取设备信息​​:需要知道手柄的VID(Vendor ID)和PID(Product ID),你可以通过设备管理器查看。在代码中可以使用这些信息来找到对应的设备。
  • ​添加引用和定义​​:在项目中添加对System.Runtime.InteropServices的引用,以便使用Windows API函数。

3. 代码实现

定义必要的结构体和API函数
using System;
using System.Runtime.InteropServices;public class HidDevice
{// HID设备信息结构体[StructLayout(LayoutKind.Sequential)]public struct HidDeviceCaps{public Int16 UsagePage;public Int16 Usage;public Int32 VersionNumber;[MarshalAs(UnmanagedType.ByValArray, SizeConst = 6)]public Int16[] LogicalMinimum;[MarshalAs(UnmanagedType.ByValArray, SizeConst = 6)]public Int16[] LogicalMaximum;[MarshalAs(UnmanagedType.ByValArray, SizeConst = 6)]public Int16[] PhysicalMinimum;[MarshalAs(UnmanagedType.ByValArray, SizeConst = 6)]public Int16[] PhysicalMaximum;public Int32 NumberOfButtons;public Int32 NumberOfValueCaps;public Int32 NumberOfDataIndices;[MarshalAs(UnmanagedType.ByValArray, SizeConst = 32)]public byte[] Data;}// 导入Windows API函数[DllImport("hid.dll")]public static extern Boolean HidD_GetHidGuid(out Guid guid);[DllImport("hid.dll")]public static extern Boolean HidD_GetDeviceCaps(IntPtr hidDeviceObject, ref HidDeviceCaps capabilities);[DllImport("hid.dll")]public static extern Boolean HidD_GetInputReport(IntPtr hidDeviceObject, IntPtr buffer, Int32 bufferSize);[DllImport("kernel32.dll")]public static extern IntPtr CreateFile(String lpFileName, Int32 dwDesiredAccess, Int32 dwShareMode, IntPtr lpSecurityAttributes, Int32 dwCreationDisposition, Int32 dwFlagsAndAttributes, IntPtr hTemplateFile);[DllImport("kernel32.dll")]public static extern Boolean CloseHandle(IntPtr hObject);[DllImport("kernel32.dll", SetLastError = true)]public static extern Boolean ReadFile(IntPtr hFile, IntPtr buffer, uint nNumberOfBytesToRead, out uint lpNumberOfBytesRead, IntPtr lpOverlapped);
}
实现HID设备的数据读取
public class HidDeviceReader
{private IntPtr _hidDeviceHandle;public HidDeviceReader(string devicePath){// 打开HID设备_hidDeviceHandle = CreateFile(devicePath, FileAccess.ReadWrite, FileShare.None, IntPtr.Zero, FileMode.Open, 0, IntPtr.Zero);if (_hidDeviceHandle.ToInt32() == -1){// 处理错误throw new Exception("Unable to open the HID device.");}}public void ReadData(byte[] buffer){// 读取HID设备数据uint bytesRead;ReadFile(_hidDeviceHandle, buffer, (uint)buffer.Length, out bytesRead, IntPtr.Zero);}public void Close(){// 关闭HID设备if (_hidDeviceHandle != IntPtr.Zero){CloseHandle(_hidDeviceHandle);_hidDeviceHandle = IntPtr.Zero;}}
}
使用HidDeviceReader类来读取数据
class Program
{static void Main(){// 这里需要知道你的HID设备的路径,格式为 @"\\.\VID_XXXX&PID_XXXX"string hidDevicePath = @"\\.\VID_XXXX&PID_XXXX";HidDeviceReader reader = new HidDeviceReader(hidDevicePath);try{// 根据你的HID设备的数据报告大小创建缓冲区,示例大小为64byte[] buffer = new byte[64](@ref);// 读取数据reader.ReadData(buffer);// 处理读取到的数据ParseJoystickData(buffer);}finally{// 确保关闭设备句柄reader.Close();}}static void ParseJoystickData(byte[] data){// 这里需要根据手柄的报告描述符来解析摇杆按键数据// 例如,假设摇杆X轴数据在data[0](@ref),Y轴数据在data[1](@ref)// 按键状态在data[2](@ref)的低8位int xValue = data[0](@ref);int yValue = data[1](@ref);byte buttonState = data[2](@ref);Console.WriteLine($"X轴位置: {xValue}");Console.WriteLine($"Y轴位置: {yValue}");Console.WriteLine($"按键状态: {buttonState}");}
}

4. 注意事项

  • ​设备路径​​:需要根据实际情况修改hidDevicePath,你可以在设备管理器中找到手柄对应的设备,其路径格式一般为@"\\.\VID_XXXX&PID_XXXX",其中XXXX是十六进制的VID和PID。
  • ​数据报告大小​​:buffer数组的大小需要根据手柄的数据报告大小来确定,不同的手柄可能不同。
  • ​数据解析​​:ParseJoystickData方法中的数据解析逻辑需要根据手柄的报告描述符来编写,不同的手柄数据格式可能不同。你可以参考HID键盘对照表等相关文档来了解数据格式。

http://www.xdnf.cn/news/2813.html

相关文章:

  • Ubuntu 20.04 安装 ROS 2 Foxy Fitzroy
  • xilinx的XCI文件设定输出目录
  • MIT XV6 - 1.1 Lab: Xv6 and Unix utilities - sleep 是怎样练成的?
  • [AI]browser-use + web-ui 大模型实现自动操作浏览器
  • 元宇宙2.0:当区块链成为数字世界的宪法
  • 【C++初阶】--- 模板进阶
  • (三十二)Android开发中AppCompatActivity和Activity之间的详细区别
  • 01_微服务常见问题
  • 如何利用Rust提升Linux服务器效率(详细操作指南)
  • dma_request_slave_channel_compat 与 dma_request_channel 的区别
  • 【C语言操作符详解(二)】--结构成员访问操作符,操作符的属性,表达式求值
  • springboot中有关数据库信息转换的处理
  • __VUE_PROD_HYDRAION_MISMATCH_DETAILS__在vue.config.js怎么配置
  • 外部存储器接口:EMIF总线
  • Jetson Xavier NX EMMC版本刷机
  • 机器人--相机
  • 【MCP Node.js SDK 全栈进阶指南】高级篇(4):自定义传输层开发
  • 前端在平常的开发中高度还原ui图的思考规范
  • 阿里开源Qwen3:大语言模型的新突破
  • AI驱动软件工程:SoftEngine 方法论与 Lynx 平台实践分析
  • 「Mac畅玩AIGC与多模态08」开发篇04 - 基于 OpenAPI Schema 开发专用 Agent 插件
  • MySQL 8.4.4 安全升级指南:从漏洞修复到版本升级全流程解析
  • Spring MVC中自定义日期类型格式转换器
  • 单片机-89C51部分:8、定时器
  • 6.3 数据分析与决策支持:数据洞察生成与决策辅助系统
  • 机器学习实操 第一部分 机器学习基础 第6章 决策树
  • jmeter-Beashell获取http请求体json
  • 在K8S迁移节点kubelet数据存储目录
  • 道德经解读分析
  • Android 进阶开发:深入掌握 ProgressBar 的使用与高级技巧