hal库 stm32 按键单击、双击、三击、四击以及长按的实现与使用

对于cubemx的配置

打开一个定时器,打开对应的按键中断即可。

代码

KeyMode.h

#ifndef __KEY_MODE_H_
#define __KEY_MODE_H_

#include "stdint.h"

#define KeyTim 	 htim7
#define KeyMode  TIM7

#endif /*__KEY_MODE_H_*/

KeyMode.c

#include "tim.h"
#include "gpio.h"
#include "KeyMode.h"

/*handle end*/

/*Private variables start*/
//bit0~3是按键PA0 的标志位
//bit4~7是按键PC13的标志位
//第0 位是按键第一次按下的标志位
//第1 位是按键第二次按下的标志位
//第2 位是按键第二次按下的标志位
//第3 位是按键第二次按下的标志位

static uint8_t KeyMode_Flag=0;

/*Private variables end*/

void singleClick_Key1Fun(void);	//按键1单击处理函数
void singleClick_Key2Fun(void);	//按键2单击处理函数

void doubleClick_Key1Fun(void);	//按键1双击处理函数
void doubleClick_Key2Fun(void);	//按键2双击处理函数

void longTerm_Key1Fun(void);	//按键1长时间按下处理函数
void longTerm_Key2Fun(void); //按键2长时间按下处理函数


void yxy_delay_us(uint32_t us)
{
		SysTick->CTRL =0;	
		SysTick->LOAD =72*us-1;
		SysTick->VAL =0;	
		SysTick->CTRL =5;	
		while((SysTick->CTRL & 0x10000) == 0);
	SysTick->CTRL =0;	
}

void yxy_delay_ms(uint32_t ms)
{
	
	while(ms--)
	{
		SysTick->CTRL =0;	
		SysTick->LOAD =72000-1;
		SysTick->VAL =0;	
		SysTick->CTRL =5;	
		while((SysTick->CTRL & 0x10000) == 0);
		
	}
	SysTick->CTRL =0;	
}

/*单击*/
void singleClick_Key1Fun(void)
{
	
}

/*双击*/
void doubleClick_Key1Fun(void)
{
	
}

/*三击*/
void threeClick_Key1Fun(void)
{
	
}

/*四击*/
void fourClick_Key1Fun(void)
{
	
}

/*长按*/
void longTerm_Key1Fun(void)
{
	
}



/* 流程

按键先按下,在300ms内没有第二次按下就代表是单击,
一但在300ms以内按下第二次就代表是双击,
当按键一直未松手,超过1500ms时就是长按。

*/



void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
{
	printf("EX\r\n");
  if(GPIO_Pin == GPIO_PIN_0)  //PA0
  {		
		if((KeyMode_Flag&0x08) == 0x08)	//判断按键1在之前是否按下3次
		{			
			KeyMode_Flag |= 0x08;//代表第四次按下了
		}		
		
		if((KeyMode_Flag&0x02) == 0x02)	//判断按键1在之前是否按下2次
		{			
			KeyMode_Flag |= 0x04;//代表第三次按下了
		}
		
		if((KeyMode_Flag&0x01) == 0x01)	//判断按键1在之前是否按下
		{			
			KeyMode_Flag |= 0x02;//代表第二次按下了
		}
		
		KeyMode_Flag |= 0x01;	//按键1按下标志位置位
		
		// 计数器值清零,,从当前开始计数,也充当了短暂的按键消抖的作用
    __HAL_TIM_SET_COUNTER(&KeyTim, 0);   
  }
}

void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)	//TIM7
{
	static uint8_t Key1_Cnt30 = 0;//判断单击还是双击的变量

	uint8_t long_termK1=0;	//判断长按的还是其他的变量
	
  if(htim->Instance == KeyMode)	//来只定时器6的溢出中断
	{		
		if((KeyMode_Flag&0x01) == 0x01)	//按键1按下计时
		{			
			while(HAL_GPIO_ReadPin(GPIOA, GPIO_PIN_0) == GPIO_PIN_SET)	//这里代表按键还未松开(对于野火来说按下时是高电平)
			{
				if(((KeyMode_Flag&0x02) == 0x02) || ((KeyMode_Flag&0x04) == 0x04) || ((KeyMode_Flag&0x08) == 0x08))//当确认是第二次以上按下的时候,直接跳出长时间按下处理函数
				{
					break;
				}
				
				yxy_delay_ms(10);//计时
								
				long_termK1++;
				if(long_termK1>=100) // 10ms*100 == 1000ms
				{
					long_termK1=0;
					KeyMode_Flag = 0;	//清空标志位
					longTerm_Key1Fun();//处理长时间按下的函数
					break;
				}
			}
			Key1_Cnt30++;
			
			
			if(Key1_Cnt30>=50 && (KeyMode_Flag&0x08) == 0x08)//四击
			{
				fourClick_Key1Fun();//四击处理函数
				Key1_Cnt30=0;
				KeyMode_Flag = 0;//清空标志位
			}
			
			if(Key1_Cnt30>=40 && (KeyMode_Flag&0x08) != 0x08)//三击
			{
				threeClick_Key1Fun();//三击处理函数
				Key1_Cnt30=0;
				KeyMode_Flag = 0;//清空标志位
			}
			
			if(Key1_Cnt30>=30 && (KeyMode_Flag&0x04) != 0x04)
			{
				if((KeyMode_Flag&0x02) == 0x02)//代表在一定的时间内,连续按下两次了,说明是双击
				{
					doubleClick_Key1Fun();//双击处理函数
				}
				else	//说明是单击
				{
					singleClick_Key1Fun();//单击处理函数
				}
				
				Key1_Cnt30=0;
				KeyMode_Flag = 0;//清空标志位
			}
		}
	}
}

使用

只要更改KeyMode.c的这些回调函数,里面的内容就可以实现功能

/*单击*/
void singleClick_Key1Fun(void)
{
	
}

/*双击*/
void doubleClick_Key1Fun(void)
{
	
}

/*三击*/
void threeClick_Key1Fun(void)
{
	
}

/*四击*/
void fourClick_Key1Fun(void)
{
	
}

/*长按*/
void longTerm_Key1Fun(void)
{
	
}

STM32按键单击双击按时检测通常需要结合中断和定时器功能。这里简述一下基本步骤,并给出一个大致的伪代码框架,实际编码需配合HALAPI: 1. 首先,在初始化阶段,配置按键中断: ```c void EXTI_Config(uint8_t keyPin) { GPIO_InitTypeDef GPIO_InitStruct; EXTI_InitTypeDef EXTI_InitStruct; // 配置GPIO输入模式 GPIO_InitStruct.Pin = keyPin; GPIO_InitStruct.Mode = GPIO_MODE_IT_FALLING; GPIO_InitStruct.Pull = GPIO_NOPULL; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); // 配置EXTI线 EXTI_InitStruct.Pin = keyPin; EXTI_InitStruct Line = EXTI_Line0; // 替换为你的按键对应的EXTI线 EXTI_InitStruct.Mode = EXTI_Mode Falling; EXTI_InitStruct.ActiveState = EXTI_ActiveHigh; EXTI_InitStruct.Interrupt = ENABLE; HAL_EXTI_Init(&EXTI_InitStruct); } ``` 2. 然后,设置定时器来检测按时限: ```c Timer_HandleTypeDef timerHandle; void TimerConfig(uint32_t timeout) { TIM_TimeBaseInitTypeDef.TIM_Prescaler = ...; // 根据系统频率计算预分频值 TIM_TimeBaseInitTypeDef.TIM_CounterMode = TIM_COUNTERMODE_UP; TIM_TimeBaseInitTypeDef.TIM_Period = timeout - 1; if (HAL_TIM_Base_Init(&timerHandle) != HAL_OK) { // 错误处理 } HAL_TIM_Base_Start_IT(&timerHandle); // 启动定时器 } ``` 3. 接下来是按键事件处理函数,它会检查是否是单击双击还是按时限: ```c static uint32_t lastPressTime = 0; // 上一次按键按下时间 static bool isLongPress = false; void Key_IRQHandler(void) { if (HAL_GPIO_ReadPin(GPIOA, keyPin)) { // 按键释放 uint32_t currentTime = HAL_GetTick(); if (currentTime - lastPressTime < 500) { // 双击检查 // 双击逻辑... isLongPress = false; } else { if (!isLongPress) { // 单击逻辑... } else { if (timeoutExpired) { // 按时达到9秒逻辑... } else { // 按时小于9秒逻辑... } } lastPressTime = currentTime; } } } ``` 这里的`500`是一个简化示例,实际双击间隔应根据应用需求调整。对于按时限,你需要在`Key_IRQHandler`中检查定时器是否超时。
评论 8
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

嵌入一下?

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值