HALSTM32通用定时器+EXTI实现单击/双击/长按功能

2023-11-11 05:04

本文主要是介绍HALSTM32通用定时器+EXTI实现单击/双击/长按功能,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

HALSTM32通用定时器+EXTI实现单击/双击/长按功能


  • ✨在使用USB功率计的时候,发现上面的一个按键实现多画面功能切换,于是探索了一下是如何实现的,将其实现的基本思路以及综合网上收集的相关实现方法,粗陋的整理了一下,将其基本功能实现了一下,具体的功能没有做,只是做了一个简单的框架。
    在这里插入图片描述
  • 👉🏻功能实现,并不能保证每次处理的按键结果,都如预期效果,主要的影响因数是单击和双击,之间的间隔时间,每个人的手感或者每次按下的时间差异并不能像计算器那么准确和标准。

🛠HAL STM32CubeMX工程配置

  • 🌿基于STM32F103芯片。

  • 🌿主要配置一个外部中断按键引脚。
    在这里插入图片描述

  • 🔖GPIO引脚按键外部使用电路参考:
    在这里插入图片描述

  • 🌿配置一个定时器
    在这里插入图片描述

  • 🌿NVIC中断使能以及优先级配置。
    在这里插入图片描述

  • 🌿其他串口和I2C配置用于显示和调试输出。(这里根据个人需求配置,个人推荐还是使用软件I2C实现比较好,硬件I2C如果程序比较复杂,cpu处理各种中断,容易导致硬件I2C出错,导致显示画面异常的情况。)
    在这里插入图片描述
    在这里插入图片描述

  • 🌿其他时钟配置和工程生成就不展示了,根据个人情况配置。

⛳功能实现

  • 🌿完善按键中断回调内容
//重写回调函数
void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
{/*判断是中断引脚*/if(GPIO_Pin == KEY_Pin){Trg++;HAL_GPIO_TogglePin(GPIOE, GPIO_PIN_5);//状态翻转}
}
  • 🌿定时器中断回调函数
void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef* htim)
{static uint32_t click = 0;static char keyStatus = 0;static char LongPress = 0;if(htim ->Instance == TIM6){if(++cnt_1ms >= 1000){cnt_1ms = 0;HAL_GPIO_TogglePin(GPIOB, GPIO_PIN_5);//状态翻转}if(Trg > 0){keyStatus = HAL_GPIO_ReadPin(KEY_GPIO_Port, KEY_Pin);//读取按键状态click++;if((click > _debounce_ms) && (keyStatus == 1) && (LongPress == 0)) //1000{if((Trg == 1) || (Trg > 2)) //单击{Trg = 0;funIndex ++;printf("_click_ms,%d\r\n", funIndex);click = 0;//清零}if(Trg == 2) //双击{Trg = 0;funIndex += 2;printf("_doubleClick %d\r\n", funIndex);click = 0;//清零}}if((click > _press_ms) && (keyStatus == 0) && (LongPress == 0)) //1000{printf("_longPressStart\r\n");LongPress = 1;}if((LongPress == 1) && (keyStatus == 1)){LongPress = 0;Trg = 0;funIndex += 3;//松开才赋值click = 0;printf("_longPressStop %d\r\n", funIndex);}}if(funIndex > 8)funIndex = 0;}
}
  • 🌿oled驱动显示内容,可以根据个人使用的屏幕规格型号自行完善,个人使用的是SH1106 1.3“寸的OLED屏幕作为显示。
  • 🌿按键菜单显示内容参考网上的内容。
unsigned char funIndex = 0;//9和0->1-1
void menu11(void);
void menu12(void);
void menu21(void);
void menu22(void);
void menu23(void);
void menu31(void);
void menu32(void);
void menu33(void);
void menu34(void);
//定义按键操作数据
KEY_TABLE table[9] =
{{0, 0, 1, 0, 2, (*menu11)},{1, 0, 1, 1, 4, (*menu12)},{2, 2, 3, 0, 5, (*menu21)},{3, 2, 3, 0, 7, (*menu22)},{4, 4, 4, 1, 4, (*menu23)},{5, 5, 6, 2, 5, (*menu31)},{6, 5, 6, 2, 6, (*menu32)},{7, 7, 8, 3, 7, (*menu33)},{8, 7, 8, 3, 8, (*menu34)},
};void ShowMenu(int16_t x, int16_t y,char *text)
{OLED_ShowString(x, y, text, 12);//x,y,字符串,字体大小
}//一级菜单1
void menu11(void)
{//    OLED_Clear();OLED_ShowString(36, 6, "menu1-1", 16);
}//一级菜单2
void menu12(void)
{
//    OLED_Clear();OLED_ShowString(36, 6, "menu1-2", 16);
}//二级菜单1
void menu21(void)
{
//    OLED_Clear();OLED_ShowString(36, 6, "menu2-1", 16);
}//二级菜单2
void menu22(void)
{
//    OLED_Clear();OLED_ShowString(36, 6, "menu2-2", 16);
}//二级菜单3
void menu23(void)
{
//    OLED_Clear();OLED_ShowString(36, 6, "menu2-3", 16);}//三级菜单1
void menu31(void)
{
//    OLED_Clear();OLED_ShowString(36, 6, "menu3-1", 16);
}//三级菜单2
void menu32(void)
{
//    OLED_Clear();OLED_ShowString(36, 6, "menu3-2", 16);
}//三级菜单3
void menu33(void)
{
//    OLED_Clear();OLED_ShowString(36, 6, "menu3-3", 16);
}//三级菜单4
void menu34(void)
{
//    OLED_Clear();OLED_ShowString(36, 6, "menu3-4", 16);}
  • 📝main主函数
int main(void)
{/* USER CODE BEGIN 1 *//* USER CODE END 1 *//* MCU Configuration--------------------------------------------------------*//* Reset of all peripherals, Initializes the Flash interface and the Systick. */HAL_Init();/* USER CODE BEGIN Init *//* USER CODE END Init *//* Configure the system clock */SystemClock_Config();/* USER CODE BEGIN SysInit *//* USER CODE END SysInit *//* Initialize all configured peripherals */MX_GPIO_Init();MX_I2C1_Init();MX_USART1_UART_Init();MX_I2C2_Init();MX_TIM6_Init();/* USER CODE BEGIN 2 */OLED_Init();HAL_TIM_Base_Start_IT(&htim6);ShowMenu(2, 0, "OneKey Mul-Menu");/* USER CODE END 2 *//* Infinite loop *//* USER CODE BEGIN WHILE */while(1){/* USER CODE END WHILE *//* USER CODE BEGIN 3 */current = table[funIndex].operation;//根据需要获取对应需要执行的函数(*current)();//执行获取到的函数}/* USER CODE END 3 */
}

📚工程源码

链接:https://pan.baidu.com/s/1eR14Lpfjqm7nA0lvwbv5xg 
提取码:vs7f

这篇关于HALSTM32通用定时器+EXTI实现单击/双击/长按功能的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/387731

相关文章

golang中slice扩容的具体实现

《golang中slice扩容的具体实现》Go语言中的切片扩容机制是Go运行时的一个关键部分,它确保切片在动态增加元素时能够高效地管理内存,本文主要介绍了golang中slice扩容的具体实现,感兴趣... 目录1. 切片扩容的触发append 函数的实现2. runtime.growslice 函数gro

golang实现动态路由的项目实践

《golang实现动态路由的项目实践》本文主要介绍了golang实现动态路由项目实践,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习... 目录一、动态路由1.结构体(数据库的定义)2.预加载preload3.添加关联的方法一、动态路由1

使用Python实现调用API获取图片存储到本地的方法

《使用Python实现调用API获取图片存储到本地的方法》开发一个自动化工具,用于从JSON数据源中提取图像ID,通过调用指定API获取未经压缩的原始图像文件,并确保下载结果与Postman等工具直接... 目录使用python实现调用API获取图片存储到本地1、项目概述2、核心功能3、环境准备4、代码实现

MySQL数据库实现批量表分区完整示例

《MySQL数据库实现批量表分区完整示例》通俗地讲表分区是将一大表,根据条件分割成若干个小表,:本文主要介绍MySQL数据库实现批量表分区的相关资料,文中通过代码介绍的非常详细,需要的朋友可以参考... 目录一、表分区条件二、常规表和分区表的区别三、表分区的创建四、将既有表转换分区表脚本五、批量转换表为分区

Spring Boot 整合 Redis 实现数据缓存案例详解

《SpringBoot整合Redis实现数据缓存案例详解》Springboot缓存,默认使用的是ConcurrentMap的方式来实现的,然而我们在项目中并不会这么使用,本文介绍SpringB... 目录1.添加 Maven 依赖2.配置Redis属性3.创建 redisCacheManager4.使用Sp

Kali Linux安装实现教程(亲测有效)

《KaliLinux安装实现教程(亲测有效)》:本文主要介绍KaliLinux安装实现教程(亲测有效),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、下载二、安装总结一、下载1、点http://www.chinasem.cn击链接 Get Kali | Kal

C#使用MQTTnet实现服务端与客户端的通讯的示例

《C#使用MQTTnet实现服务端与客户端的通讯的示例》本文主要介绍了C#使用MQTTnet实现服务端与客户端的通讯的示例,包括协议特性、连接管理、QoS机制和安全策略,具有一定的参考价值,感兴趣的可... 目录一、MQTT 协议简介二、MQTT 协议核心特性三、MQTTNET 库的核心功能四、服务端(BR

SpringCloud整合MQ实现消息总线服务方式

《SpringCloud整合MQ实现消息总线服务方式》:本文主要介绍SpringCloud整合MQ实现消息总线服务方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录一、背景介绍二、方案实践三、升级版总结一、背景介绍每当修改配置文件内容,如果需要客户端也同步更新,

Dubbo之SPI机制的实现原理和优势分析

《Dubbo之SPI机制的实现原理和优势分析》:本文主要介绍Dubbo之SPI机制的实现原理和优势,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Dubbo中SPI机制的实现原理和优势JDK 中的 SPI 机制解析Dubbo 中的 SPI 机制解析总结Dubbo中

使用Java实现Navicat密码的加密与解密的代码解析

《使用Java实现Navicat密码的加密与解密的代码解析》:本文主要介绍使用Java实现Navicat密码的加密与解密,通过本文,我们了解了如何利用Java语言实现对Navicat保存的数据库密... 目录一、背景介绍二、环境准备三、代码解析四、核心代码展示五、总结在日常开发过程中,我们有时需要处理各种软