STM32蓝牙HID实战:打造低功耗、高性能的客制化键盘

news2025/1/13 13:51:05

一、项目概述

本项目旨在使用STM32单片机打造一款功能强大的蓝牙客制化键盘,它拥有以下特点:

  • 九键布局,小巧便携: 满足日常使用需求,方便携带。
  • 全键可编程: 所有按键和旋钮均可通过电脑软件自定义快捷键,实现个性化功能。
  • 蓝牙无线连接: 摆脱线缆束缚,提供更自由的使用体验。

二、硬件设计

2.1 硬件平台
  • 主控芯片: STM32F103C8T6
  • 蓝牙模块: HC-05
  • 按键: 机械轴*9
  • 旋钮编码器: EC11
  • 其他: 电焊、杜邦线、PCB板等
2.2 电路原理图

2.3 PCB设计
  • 使用KiCad等EDA软件进行PCB设计,确保电路稳定可靠。
  • 采用合理的布局,优化空间利用率,打造紧凑的外观。

三、软件设计

3.1 开发环境
  • IDE: Keil MDK
  • 编译器: ARMCC
  • 调试器: ST-Link
3.2 软件架构

3.3 代码实现

3.3.1 蓝牙初始化

void Bluetooth_Init(void)
{
  // 设置蓝牙模块波特率为9600
  USART1_Init(9600);

  // 发送AT指令进入AT模式
  USART1_SendString("AT\r\n");
  // 设置蓝牙模块名称
  USART1_SendString("AT+NAME=CustomKeyboard\r\n");
  // 设置蓝牙模块配对密码
  USART1_SendString("AT+PIN=1234\r\n");
  // 设置蓝牙模块为从模式
  USART1_SendString("AT+ROLE=0\r\n");
  // 开启蓝牙模块
  USART1_SendString("AT+CMODE=1\r\n");
}

代码解释:

  • 这部分代码首先初始化了STM32的USART1,用于与HC-05蓝牙模块通信。
  • 随后,代码发送一系列AT指令配置蓝牙模块:
    • AT: 测试指令,确保蓝牙模块连接正常。
    • AT+NAME=CustomKeyboard: 设置蓝牙模块名称为 "CustomKeyboard"。
    • AT+PIN=1234: 设置蓝牙模块配对密码为 "1234"。
    • AT+ROLE=0: 将蓝牙模块设置为从模式,等待连接。
    • AT+CMODE=1: 允许蓝牙模块连接任何地址的设备。

3.3.2 按键扫描

uint8_t KeyScan(void)
{
  // 扫描按键矩阵
  // ...
  // 返回按键值
  return key_value;
}

代码解释:

  • 这段代码是按键扫描函数的框架。你需要根据你的硬件电路实现具体的按键扫描逻辑。
  • 一般来说,你需要使用GPIO模拟矩阵键盘的扫描方式,检测哪个按键被按下。
  • 函数最后需要返回被按下的按键码,如果没有按键按下则返回0。

3.3.3 旋钮读取

int8_t Encoder_Read(void)
{
  static uint8_t last_state = 0;
  uint8_t current_state = (GPIOB->IDR & 0x03); // 读取A、B相电平

  if (current_state != last_state) {
    if ((current_state == 0x01 && last_state == 0x03) ||
        (current_state == 0x03 && last_state == 0x02) ||
        (current_state == 0x02 && last_state == 0x00) ||
        (current_state == 0x00 && last_state == 0x01)) {
      return 1; // 顺时针旋转
    } else {
      return -1; // 逆时针旋转
    }
  }
  last_state = current_state;
  return 0; // 未旋转
}

代码解释:

  • 这段代码实现了读取旋转编码器数值的逻辑。
  • 它首先读取编码器的A、B两相的电平状态。
  • 然后通过对比当前状态和上次状态,判断编码器的旋转方向。
  • 如果顺时针旋转,返回1;逆时针旋转,返回-1;没有旋转,返回0。

3.3.4 数据处理

  • 键盘使用特定的数据格式将按键信息和旋钮信息发送给电脑:

    • 第一个字节代表数据类型:
      • 0x01:代表按键按下/弹起事件。
      • 0x02:代表旋钮旋转事件。
    • 第二个字节代表按键码或旋钮方向:
      • 对于按键事件,该字节表示被按下或弹起的按键的键码。
      • 对于旋钮事件,该字节为 0x00 表示逆时针旋转, 0x01 表示顺时针旋转。
  • 定义按键码:

#define KEY_1 0x01
#define KEY_2 0x02
// ...
#define KEY_9 0x09
  • 数据打包:
uint8_t data_buffer[2];

void Data_Process(uint8_t key_value, int8_t encoder_value) {
  if (key_value != 0) {
    // 处理按键事件
    data_buffer[0] = 0x01; // 数据类型:按键
    data_buffer[1] = key_value; // 按键码
  } else if (encoder_value != 0) {
    // 处理旋钮事件
    data_buffer[0] = 0x02; // 数据类型:旋钮
    data_buffer[1] = (encoder_value > 0) ? 0x01 : 0x00; // 旋转方向
  }
}

3.3.5 蓝牙发送

void Bluetooth_Send(uint8_t *data, uint8_t len) {
  // 通过蓝牙串口发送数据
  for (uint8_t i = 0; i < len; i++) {
    USART1_SendByte(data[i]);
  }
}

代码解释:

  • 这段代码实现了通过蓝牙串口发送数据的函数。
  • 它接受一个指向数据缓冲区的指针 data 和数据的长度 len 作为参数。
  • 函数内部使用循环遍历数据缓冲区,并将每个字节数据通过 USART1_SendByte 函数发送出去。

代码实例:

// 假设 data_buffer 已经填充了要发送的数据
uint8_t data_buffer[2] = {0x01, 0x03}; // 例如:按键事件,按键码为 KEY_3

// 通过蓝牙发送数据
Bluetooth_Send(data_buffer, sizeof(data_buffer)); 

完整代码示例:

// ... 其他代码 ...

// 蓝牙发送函数
void Bluetooth_Send(uint8_t *data, uint8_t len) {
  // 通过蓝牙串口发送数据
  for (uint8_t i = 0; i < len; i++) {
    USART1_SendByte(data[i]);
  }
}

// 主函数
int main(void) {
  // ... 初始化代码 ...

  while (1) {
    // 扫描按键
    uint8_t key_value = KeyScan();

    // 读取旋钮状态
    int8_t encoder_value = Encoder_Read();

    // 处理数据
    Data_Process(key_value, encoder_value);

    // 如果有数据需要发送
    if (data_buffer[0] != 0) {
      // 通过蓝牙发送数据
      Bluetooth_Send(data_buffer, sizeof(data_buffer));

      // 清空数据缓冲区
      data_buffer[0] = 0; 
    }
  }
}

注意:

  • 你需要根据你的硬件电路和数据协议,修改 KeyScan, Encoder_Read 和 Data_Process 函数的具体实现。
  • 你需要将 USART1_SendByte 函数替换为你实际使用的串口发送函数。

四、电脑端软件

为了实现自定义快捷键功能,你需要开发一个电脑端软件,该软件需要实现以下功能:

  1. 连接蓝牙键盘: 搜索并连接你的蓝牙键盘设备。
  2. 接收数据: 持续接收来自蓝牙键盘的数据。
  3. 解析数据: 根据预定义的数据格式解析接收到的数据,识别按键事件和旋钮事件。
  4. 执行快捷键: 根据用户预先设置的快捷键映射关系,执行相应的操作。例如,用户可以将 KEY_1 映射为 Ctrl+C 快捷键,将旋钮顺时针旋转映射为 音量+ 操作。

以下是一个使用 Python 实现的电脑端软件示例代码:

import bluetooth
import keyboard  # 需要安装 keyboard 库: pip install keyboard

# 蓝牙键盘设备地址
BT_ADDR = "00:11:22:33:44:55"
# 蓝牙服务UUID
BT_UUID = "00001124-0000-1000-8000-00805F9B34FB"

def handle_data(data):
    """处理接收到的数据"""
    data_type = data[0]
    data_value = data[1]

    if data_type == 0x01:  # 按键事件
        key_code = data_value
        print(f"按键事件: {key_code}")
        # TODO: 根据 key_code 执行相应的快捷键操作
    elif data_type == 0x02:  # 旋钮事件
        direction = "顺时针" if data_value == 0x01 else "逆时针"
        print(f"旋钮事件: {direction}")
        # TODO: 根据 direction 执行相应的操作

def main():
    """主函数"""
    print("正在搜索蓝牙设备...")
    devices = bluetooth.discover_devices(lookup_names=True)
    for addr, name in devices:
        if addr == BT_ADDR:
            print(f"找到设备: {name} ({addr})")
            break
    else:
        print("未找到设备")
        return

    print("正在连接...")
    sock = bluetooth.BluetoothSocket(bluetooth.RFCOMM)
    sock.connect((BT_ADDR, 1))  # 假设蓝牙服务端口号为 1
    print("连接成功")

    try:
        while True:
            data = sock.recv(1024)
            if data:
                handle_data(data)
    except KeyboardInterrupt:
        print("程序退出")
    finally:
        sock.close()

if __name__ == "__main__":
    main()

代码说明:

  1. 导入库: 导入 bluetooth 库用于蓝牙通信,导入 keyboard 库用于模拟键盘操作。
  2. 定义常量: 定义蓝牙键盘的设备地址 BT_ADDR 和服务 UUID BT_UUID
  3. handle_data() 函数: 该函数用于处理接收到的数据,根据数据类型和数据值执行相应的操作。
  4. main() 函数: 该函数是程序的入口点,负责搜索蓝牙设备、连接设备、接收数据并调用 handle_data() 函数处理数据。
  5. 模拟快捷键: 在 handle_data() 函数中,你可以使用 keyboard 库提供的函数模拟键盘操作来实现快捷键功能。例如,使用 keyboard.press_and_release('ctrl+c') 模拟 Ctrl+C 快捷键。

注意:

  • 你需要将 BT_ADDR 替换为你的蓝牙键盘的实际地址。
  • 你需要根据你的键盘硬件和数据协议修改代码。
  • 你需要根据你的需求修改 handle_data() 函数中的快捷键映射关系。

五、总结

本文介绍了如何使用STM32制作一款蓝牙客制化键盘,并详细讲解了硬件设计、软件设计以及数据传输协议等方面的内容。通过该项目,你可以学习到蓝牙通信、按键扫描、编码器读取等知识,并锻炼嵌入式系统开发能力。

你可以根据自己的需求,进一步扩展键盘的功能,例如增加RGB背光、支持多层配置、实现宏定义等。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1906392.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

LLM推理引擎怎么选?TensorRT vs vLLM vs LMDeploy vs MLC-LLM

LLM擅长文本生成应用程序&#xff0c;如聊天和代码完成模型&#xff0c;能够高度理解和流畅。但是它们的大尺寸也给推理带来了挑战。有很多个框架和包可以优化LLM推理和服务&#xff0c;所以在本文中我将整理一些常用的推理引擎并进行比较。 TensorRT-LLM TensorRT-LLM是NV发布…

【数据结构】(C语言):二叉搜索树(不使用递归)

二叉搜索树&#xff1a; 非线性的&#xff0c;树是层级结构。基本单位是节点&#xff0c;每个节点最多2个子节点。有序。每个节点&#xff0c;其左子节点都比它小&#xff0c;其右子节点都比它大。每个子树都是一个二叉搜索树。每个节点及其所有子节点形成子树。可以是空树。 …

控件-ProgressBar

常用属性 1.android:max:进度条的最大值 2. android: progress:进度条已完成进度值 3. android: indeterminate:如果设置成true,则进度条不精确显示进度 4.style"?android:attr/progressBarStyleHorizontal"水平进度条 案例 进度条加载

基于Java+SpringMvc+Vue技术的就医管理系统设计与实现系统(源码+LW+部署讲解)

目录 界面展示 第六章 部分代码实现 6.1 Spring boot 配置代码 6.2 用户管理及登录登出代码 6.3 Md5 加密算法代码 6.4 部分数据库代码 六、论文参考&#xff1a; 七、其他案例&#xff1a; 系统介绍&#xff1a; 就医管理系统&#xff0c;也称为医院管理系统&#…

STM32自己从零开始实操08:STM32主控原理图

由于老师使用的各引脚分门别类的单片机原理图我没有找到&#xff0c;我使用是引脚按顺序摆放的&#xff0c;不方便一个模块一个模块截图展示&#xff0c;所以这部分使用老师的原理图。 一、电源 1.1电源的介绍 1.1.1数字电源和地&#xff08;VDD和VSS&#xff09; 数字电源…

AIGC时代程序员的跃迁——编程高手的密码武器

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…

RocketMQ之消费者带你了解概念和消费流程

1. 背景 RocketMQ 的消费可以算是 RocketMQ 的业务逻辑中最复杂的一块。这里面涉及到许多消费模式和特性。本想一篇文章写完&#xff0c;写到后面发现消费涉及到的内容太多&#xff0c;于是决定分多篇来写。本文作为消费系列的第一篇&#xff0c;主要讲述 RocketMQ 消费涉及到…

网络规划与设计————期末复习

一、选择题&#xff08;每题1分&#xff09; 1、光纤线组建的标准以太网是______。 A.10BASE-5 B.10BASE-2 C.10BASE-T D.10BASE-F 其实也很好记&#xff0c;光纤的英文是 "Fiber Optic"&#xff0c;双绞线的英文是 "Twisted Pair"。 5呢…

Redis核心问题总结(一)

1、为什么要使用Redis做缓存 缓存的好处 使用缓存的目的就是提升读写性能。而实际业务场景下&#xff0c;更多的是为了提升读性能&#xff0c;带来更好的性 能&#xff0c;带来更高的并发量。Redis 的读写性能比 Mysql 好的多&#xff0c;我们就可以把 Mysql 中的热点数据缓 …

Python28-8 GBM梯度提升算法

梯度提升算法&#xff08;Gradient Boosting Machine&#xff0c;GBM&#xff09;是一种集成学习方法&#xff0c;通过逐步构建一系列简单模型&#xff08;通常是决策树&#xff09;&#xff0c;并结合这些模型来提高整体预测性能。GBM广泛用于回归和分类任务&#xff0c;因为它…

端到端自动驾驶新突破:Nvidia提出全并行PARA-Drive,斩获CVPR挑战赛冠军

论文标题&#xff1a; PARA-Drive: Parallelized Architecture for Real-time Autonomous Driving 论文作者&#xff1a; Xinshuo Weng, Boris Ivanovic, Yan Wang, Yue Wang, Marco Pavone 导读&#xff1a; 本文系统分析了自动驾驶高级架构的设计空间&#xff0c;提出了关…

单片机软件架构连载(3)-typedef

今天给大家讲typedef&#xff0c;这个关键字在实际产品开发中&#xff0c;也是海量应用。 技术涉及知识点比较多&#xff0c;有些并不常用&#xff0c;我们以贴近实际为原则&#xff0c;让大家把学习时间都花在重点上。 1.typedef的概念 typedef 是 C 语言中的一个关键字&…

artts升级版本后常见的编译错误(定期更新......)

1、设置泛型将参数配置为 null 时抛出了如下异常: Type null is not assignable to type T. T could be instantiated with an arbitrary type which could be unrelated to null. <ArkTSCheck> 解决办法 在 null 后面添加 ! 即可,以表示该值不会为 null data: T null!…

【可能是全网最丝滑的LangChain教程】十七、LangChain进阶之Retrievers

人生不能像做菜&#xff0c;把所有的料都准备好了才下锅。 01 Retrievers介绍 检索器&#xff08;Retrievers&#xff09; 是一种接口&#xff0c;用于根据非结构化查询返回文档&#xff0c;它比向量存储更为通用&#xff0c;既可以使用向量存储作为底层&#xff0c;也可以是其…

C++11右值引用及移动构造

区分左值和右值 在学习c11的右值引用前&#xff0c;大家肯定会有点陌生什么是右值&#xff1f;什么是左值&#xff1f;现在我先来带大家熟悉一下概念。 左值 可以被取地址&#xff0c;也可被修改&#xff08;const修饰的除外&#xff09; 可以出现在等号左边&#xff0c;也可…

华为HCIP Datacom H12-821 卷29

1.多选题 下面关于LSA age字段&#xff0c;描述正确的是∶ A、LSA age的单位为秒&#xff0c;在LSDB中的LSA的LS age随时间增长而增长 B、LSA age的单位为秒&#xff0c;在LSDB中的LSA的LS age随时间增长而减少 C、如果一条LSA的LS age达到了LS RefreshTime&#xff08…

【C++】AVL树(旋转、平衡因子)

&#x1f308;个人主页&#xff1a;秦jh_-CSDN博客&#x1f525; 系列专栏&#xff1a;https://blog.csdn.net/qinjh_/category_12575764.html?spm1001.2014.3001.5482 ​ 目录 前言 AVL树的概念 节点 插入 AVL树的旋转 新节点插入较高左子树的左侧---左左&#xff1a;…

Spring的AOP基础以及AOP的核心概念

2. AOP基础 学习完spring的事务管理之后&#xff0c;接下来我们进入到AOP的学习。 AOP也是spring框架的第二大核心&#xff0c;我们先来学习AOP的基础。 在AOP基础这个阶段&#xff0c;我们首先介绍一下什么是AOP&#xff0c;再通过一个快速入门程序&#xff0c;让大家快速体…

高级RAG检索中的五种查询重写策略_用于检索增强的大型语言模型的查询重写

一、前言 检索增强生成 (RAG) 作为人工智能 (AI) 领域的一项重要技术&#xff0c;近年来得到了飞速发展。它将基于检索模型和基于生成的模型相结合&#xff0c;利用海量外部数据&#xff0c;生成更具信息量、更准确、更具语境相关性的回复。检索策略是 RAG 系统的关键组成部分…

2024年最适合高级网工的11款Linux

号主&#xff1a;老杨丨11年资深网络工程师&#xff0c;更多网工提升干货&#xff0c;请关注公众号&#xff1a;网络工程师俱乐部 你们好&#xff0c;我的网工朋友。 Linux作为一个免费且开源的操作系统&#xff0c;随着时间的推移催生了多个发行版&#xff0c;并且得到了庞大…