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

一、项目概述

本项目旨在使用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)
{// 设置蓝牙模块波特率为9600USART1_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_valueprint(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})")breakelse:print("未找到设备")returnprint("正在连接...")sock = bluetooth.BluetoothSocket(bluetooth.RFCOMM)sock.connect((BT_ADDR, 1))  # 假设蓝牙服务端口号为 1print("连接成功")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.mzph.cn/bicheng/40248.shtml

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

相关文章

如何用java语言+若依开源框架开发一套数字化产科系统 数字化产科管理平台源码

如何用java语言若依开源框架开发一套数字化产科系统 数字化产科管理平台源码 要使用Java语言和若依&#xff08;RuoYi&#xff09;开源框架来开发一个数字化产科系统&#xff0c;你需要遵循一系列步骤&#xff0c;从环境搭建到系统设计与开发&#xff0c;再到测试与部署。 以下…

2023年问界M9 EV 问界M9增程维修手册和电路图线路图资料更新

此次更新了2023年问界M9 EV及问界M9增程维修手册和电路图资料&#xff0c;覆盖市面上99%车型&#xff0c;包括维修手册、电路图、新车特征、车身钣金维修数据、全车拆装、扭力、发动机大修、发动机正时、保养、电路图、针脚定义、模块传感器、保险丝盒图解对照表位置等等&#…

Redis的八种数据类型介绍

Redis 是一个高性能的键值存储&#xff0c;它支持多种丰富的数据类型。每种数据类型都有其特定的用途和底层实现。下面我将介绍 Redis 支持的主要数据类型及其背后的数据结构。 本人这里还有几篇详细的Redis用法文章&#xff0c;可以用来进阶康康&#xff01; 1. 字符串 (Stri…

ubuntu24.04LTS防火墙设置

Ubuntu24.04LTS开箱自带ufw&#xff0c;一定程度避免了开机下载ufw被攻击&#xff0c;excellent 转载aliyun教程 sudo ufw enbale可以启用并且开机自启(显示有效&#xff0c;未nmap实测) 教程3 转载自CSDN 完整格式如下&#xff1a; # 禁止IP连接端口 sudo ufw deny proto tc…

Cherno 游戏引擎笔记 (45~60)

有几个部分的笔记以图片形式呈现&#xff08;如果没找到文本可以查看是否遗漏了图片笔记&#xff09; My Github REPO(GitHub - JJJJJJJustin/Nut: The game_engine which learned from Cherno) 源码笔记&#xff0c;希望帮到你 :-} ---Shader Library&#xff08;着色器库&…

南京观海微电子----AC/DC、DC/DC转换器知识

什么是AC&#xff1f; Alternating Current&#xff08;交流&#xff09;的首字母缩写。 AC是大小和极性&#xff08;方向&#xff09;随时间呈周期性变化的电流。 电流极性在1秒内的变化次数被称为频率&#xff0c;以Hz为单位表示。 什么是DC? Direct Current&#xff08;直流…

visual studio远程调试

场景一&#xff08;被远程调试的电脑&#xff09; 确定系统位数 我这里是x64的 找到msvsmon.exe msvsmon.exe目录位置解释&#xff1a; “F:\App\VisualStudio\an\Common7\IDE\”是visual studio所在位置、 “Remote Debugger\”是固定位置、 “x64”是系统位数。 拼起来就是…

grid布局下的展开/收缩过渡效果【vue/已验证可正常运行】

代码来自GPT4o&#xff1a;国内官方直连GPT4o <template><div class"container"><button class"butns" click"toggleShowMore">{{ showAll ? 收回 : 显示更多 }}</button><transition-group name"slide-fade&…

数据库原理实验报告第二次-SQL Server SSMS工具创建和管理数据库及数据表.

题目 1、使用SSMS工具创建名为ecommerce的数据库&#xff0c;并查看或修改数据库属性 2、在数据库ecommerce中创建如下表&#xff1a; &#xff08;1&#xff09;商品类别表category 字段名 数据类型 允许NULL值 约束 字段说明 catno int 否 主键 商品类别编号 ca…

AI是在帮助开发者还是取代他们?

一&#xff1a;介绍 生成式人工智能&#xff08;AIGC&#xff09;在软件开发领域的应用确实为开发者带来了很多便利和效率提升。AI工具可以通过代码生成、错误检测、自动化测试等功能&#xff0c;帮助开发者更快速地开发和优化软件&#xff0c;减少重复性工作&#xff0c;提高…

哈喽GPT-4o,对GPT-4o 论文速写的思考与探索

作为一款强大的语言模型&#xff0c;ChatGPT 在论文写作上具备显著优势。它能够辅助学者或研究人员自动创建论文框架、摘要、文献综述及论文段落&#xff08;如引言、方法、结果、结论等&#xff09;。此外&#xff0c;ChatGPT 还能优化论文结构、润色、降低内容重复率&#xf…

比Proxmox VE更易用的免费虚拟化平台

之前虚拟化一直玩Proxmox VE&#xff0c;最近发现一个更易用的虚拟化软件CSYun&#xff0c;他与Proxmox VE类似&#xff0c;都是一个服务器虚拟化平台。它不像VMware ESXi那么复杂&#xff0c;对于个人使用者和中小企业是一个比较好的选择。 这个软件所在的网址为&#xff1a;…

【Python】已解决TypeError: init() got an unexpected keyword argument ‘threshold’

文章目录 一、分析问题背景二、可能出错的原因三、错误代码示例四、正确代码示例五、注意事项 已解决TypeError: init() got an unexpected keyword argument ‘threshold’ 一、分析问题背景 在Python编程中&#xff0c;遇到“TypeError: init() got an unexpected keyword …

Three.js机器人与星系动态场景(二):强化三维空间认识

在上篇博客中介绍了如何快速利用react搭建three.js平台&#xff0c;并实现3D模型的可视化。本文将在上一篇的基础上强化坐标系的概念。引入AxesHelper辅助工具及文本绘制工具&#xff0c;带你快速理解camer、坐标系、position、可视区域。 Three.js机器人与星系动态场景&#x…

色彩搭配的艺术:打造和谐视觉体验的秘诀

当设计作品呈现给用户时首先映入眼帘的是视觉表达&#xff0c;色彩无疑是最关键的元素之一。色彩不仅是视觉艺术的一部分&#xff0c;也承载着情感文化甚至个人品味的多重含义。在设计领域&#xff0c;色彩设计可以极大地影响作品的整体感受和传达效果。那么什么是色彩设计&…

python拉取gitlab项目以及拉取报错处理

问题解决 问题1、unable to access https://gitlab.dome.com/web-dome/dome.git/: SSL certificate problem: self signed certificate 解决&#xff1a;打开本地git bash输入下面代码 git config --global http.sslVerify false; 问题2、Authentication failed for https:…

ArmPiPro-网络配置

说明 因为ubuntu18.04有点旧&#xff0c;这里同时用上了netplan和nm(nmcli)这两个网络工具&#xff0c;如果是ubuntu22.04&#xff0c;网络管理全部用nmcli和nmtui即可。 eth0有线一般用来连接电脑&#xff0c;wlan0即是用来连接WiFi热点。 设置Pi4Lubuntu的Wifi模式&#xff…

ComboBox控件如果有两个相同的项,并且你选择了第二个,再次打开下拉列表时,它可能不会定位到你选择的上面第一个项问题

一、MFC的ComBox绑定项,有两个项名称相同,选择第二个绑定项,再次打开下拉框会定位到第 在MFC中&#xff0c;ComboBox控件默认会在选择项改变时发送CBN_SELCHANGE通知&#xff0c;但如果ComboBox中有两个相同的项&#xff0c;并且你选择了第二个&#xff0c;再次打开下拉列表时…

CSS动画keyframes简单样例

一、代码部分 1.html <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><link rel"stylesheet" href…

数据泄露时代的安全之道:访问认证的重要性

引言 想象一下&#xff1a;你一觉醒来&#xff0c;收到一条通知——你的公司遭遇了数据泄露。你感到恐惧&#xff0c;因为这意味着客户数据被曝光&#xff0c;公司声誉受损&#xff0c;还有巨额罚款在等着你。在当今的数字化环境中&#xff0c;这种情况太常见了。全球各地的组…