0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

基于non-OS和RT-Thread的按键库源码及应用

GReq_mcu168 来源:嵌入式大杂烩 作者:嵌入式大杂烩 2022-05-23 09:39 次阅读

按键在电子产品中很常见,今天给大家分享一套按键库源码及应用。

https://gitee.com/zhengnianli/EmbedSummary

FlexibleButton介绍

FlexibleButton 是一个基于标准 C 语言的小巧灵活的按键处理库,支持单击、连击、短按、长按、自动消抖,可以自由设置组合按键,可用于中断和低功耗场景。

该按键库解耦了具体的按键硬件结构,理论上支持轻触按键与自锁按键,并可以无限扩展按键数量。

另外,FlexibleButton 使用扫描的方式一次性读取所有所有的按键状态,然后通过事件回调机制上报按键事件。

核心的按键扫描代码仅有三行,没错,就是经典的三行按键扫描算法。使用 C 语言标准库 API 编写,也使得该按键库可以无缝兼容任意的处理器平台,并且支持任意 OS 和 non-OS(裸机编程)。

仓库链接:

https://github.com/murphyzhao/FlexibleButton

license:Apache-2.0。

同类型的按键处理库还有MultiButton:

https://github.com/0x1abin/MultiButton

FlexibleButton的使用

FlexibleButton 包含有两个文件:

flexible_button.c、flexible_button.h

使用起来很简单,作者在README中也很详细地介绍了FlexibleButton 的使用。

下面,我们基于小熊派IOT开发板来简单实践实践:基于裸机及基于RT-Thread

1、基于non-OS(裸机编程)

板子上有两个用户按键及一个用户LED

我们实现如下操作:

  • 单击button0(即F1按键),点亮led。
  • 单机button1(即F2按键),熄灭led。
  • 双击button0(即F1按键),点亮led。
  • 双击button1(即F2按键),熄灭led。
  • 同时按下button0及button1,点亮led。

FlexibleButton 给我们提供了很多按键事件给我们使用,基本涵盖了我们日常使用按键的各种场景。FlexibleButton支持的按键事件如:

左右滑动查看全部代码>>>

typedefenum
{
FLEX_BTN_PRESS_DOWN=0,//按下事件
FLEX_BTN_PRESS_CLICK,//单击事件
FLEX_BTN_PRESS_DOUBLE_CLICK,//双击事件
FLEX_BTN_PRESS_REPEAT_CLICK,//连击事件,使用flex_button_t中的click_cnt断定连击次数
FLEX_BTN_PRESS_SHORT_START,//短按开始事件
FLEX_BTN_PRESS_SHORT_UP,//短按抬起事件
FLEX_BTN_PRESS_LONG_START,//长按开始事件
FLEX_BTN_PRESS_LONG_UP,//长按抬起事件
FLEX_BTN_PRESS_LONG_HOLD,//长按保持事件
FLEX_BTN_PRESS_LONG_HOLD_UP,//长按保持的抬起事件
FLEX_BTN_PRESS_MAX,
FLEX_BTN_PRESS_NONE,
}flex_button_event_t;
这些按键事件就是FlexibleButton返回给我们应用层的,我们只要在应用层做相关的按键处理就可以。比如单击按键时,我们要做什么逻辑控制;按键双击时,又要做怎样的逻辑控制等等。

所以,哪怕我们的板子只有一两个按键,也可以做很多按键控制。

下面来一起实操一下:

首先,准备一个按键相关工程,把flexible_button.c、flexible_button.h添加到工程里。

flexible_button.h对外提供了如下几个接口

左右滑动查看全部代码>>>

int32_tflex_button_register(flex_button_t*button);//按键注册
flex_button_event_tflex_button_event_read(flex_button_t*button);//按键事件读取
uint8_tflex_button_scan(void);//按键扫描
flex_button_register用于按键注册,需要用户至少提供如下按键信息
  • 按键ID
  • 按键引脚电平读取函数
  • 事件回调函数
  • 设置按键按下的逻辑电平
  • 设置短按事件触发的起始 tick
  • 设置长按事件触发的起始 tick
  • 设置长按保持事件触发的起始 tick

flex_button_register在初始化时进行调用,如:

左右滑动查看全部代码>>>

staticvoiduser_button_init(void)
{
inti;

memset(&user_button[0],0x0,sizeof(user_button));

for(i=0;i< USER_BUTTON_MAX; i ++)
    {
        user_button[i].id = i;   //按键的ID号
user_button[i].usr_button_read=common_btn_read;//按键引脚电平读取函数
user_button[i].cb=common_btn_evt_cb;//事件回调函数
user_button[i].pressed_logic_level=0;//设置按键按下的逻辑电平
user_button[i].short_press_start_tick=FLEX_MS_TO_SCAN_CNT(1500);//设置短按事件触发的起始tick
user_button[i].long_press_start_tick=FLEX_MS_TO_SCAN_CNT(3000);//设置长按事件触发的起始tick
user_button[i].long_hold_start_tick=FLEX_MS_TO_SCAN_CNT(4500);//设置长按保持事件触发的起始tick

flex_button_register(&user_button[i]);//按键注册
}
}
这种机制很常用。

比如,一些美食教程,常常提供一些制作巧克力的方法,而不是每种口味的巧克力都教一遍,因为方法基本都差不多。不同的人喜欢不同的巧克力口味,根据自己需要,准备做巧克力的原料,再套用制作方法就可以。

FlexibleButton提供一个管理按键的框架,我们根据不同的的芯片或者不同的环境提供FlexibleButton需要的一些按键信息,就可以起到相同效果。

FlexibleButton数据结构:

左右滑动查看全部代码>>>

typedefstructflex_button
{
structflex_button*next;//按键库使用单向链表串起所有的按键

uint8_t(*usr_button_read)(void*);//用户设备的按键引脚电平读取函数,重要
flex_button_response_callbackcb;//设置按键事件回调,用于应用层对按键事件的分类处理

uint16_tscan_cnt;//用于记录扫描次数,按键按下是开始从零计数
uint16_tclick_cnt;//记录单击次数,用于判定单击、连击
uint16_tmax_multiple_clicks_interval;//连击间隙,用于判定是否结束连击计数,有默认值

uint16_tdebounce_tick;//消抖时间,暂未使用,依靠扫描间隙进行消抖
uint16_tshort_press_start_tick;//设置短按事件触发的起始tick
uint16_tlong_press_start_tick;//设置长按事件触发的起始tick
uint16_tlong_hold_start_tick;//设置长按保持事件触发的起始tick

uint8_tid;//当多个按键使用同一个回调函数时,用于断定属于哪个按键
uint8_tpressed_logic_level:1;//设置按键按下的逻辑电平
uint8_tevent:4;//用于记录当前按键事件
uint8_tstatus:3;//用于记录当前按键的状态,用于内部状态机
}flex_button_t;
按键引脚电平读取函数如:

左右滑动查看全部代码>>>

staticuint8_tcommon_btn_read(void*arg)
{
uint8_tvalue=0;

flex_button_t*btn=(flex_button_t*)arg;

switch(btn->id)
{
caseUSER_BUTTON_0:
value=HAL_GPIO_ReadPin(USER_BUTTON_0_PORT,USER_BUTTON_0_PIN);
break;
caseUSER_BUTTON_1:
value=HAL_GPIO_ReadPin(USER_BUTTON_1_PORT,USER_BUTTON_1_PIN);
break;
default:
assert_param(0);
}

returnvalue;
}
按键事件回调函数如:

左右滑动查看全部代码>>>

//按键事件回调函数
staticvoidcommon_btn_evt_cb(void*arg)
{
flex_button_t*btn=(flex_button_t*)arg;

//非组合按键事件处理
non_combination_btn_event(btn);

//组合按键事件处理
if((flex_button_event_read(&user_button[USER_BUTTON_0])==FLEX_BTN_PRESS_CLICK)&&
(flex_button_event_read(&user_button[USER_BUTTON_1])==FLEX_BTN_PRESS_CLICK))
{
printf("[combination]:button0andbutton1,LEDON>>>>>>>>>>>>>>>>>>>>>>>
");
HAL_GPIO_WritePin(LED_GPIO_Port,LED_Pin,GPIO_PIN_SET);//亮
}
}

//非组合按键事件处理
staticvoidnon_combination_btn_event(flex_button_t*btn)
{
switch(btn->id)
{
caseUSER_BUTTON_0:
{
switch(btn->event)
{
caseFLEX_BTN_PRESS_DOWN:
printf("%s:%s
",enum_btn_id_string[btn->id],enum_event_string[btn->event]);
break;
caseFLEX_BTN_PRESS_CLICK:
HAL_GPIO_WritePin(LED_GPIO_Port,LED_Pin,GPIO_PIN_SET);//亮
printf("%s:%s
",enum_btn_id_string[btn->id],enum_event_string[btn->event]);
printf("<<<<<<<<<<<<<<<<<<<<<<<;
caseFLEX_BTN_PRESS_DOUBLE_CLICK:
HAL_GPIO_WritePin(LED_GPIO_Port,LED_Pin,GPIO_PIN_SET);//亮
printf("%s:%s
",enum_btn_id_string[btn->id],enum_event_string[btn->event]);
printf("<<<<<<<<<<<<<<<<<<<<<<<;
default:
break;
}
break;
}

caseUSER_BUTTON_1:
{
switch(btn->event)
{
caseFLEX_BTN_PRESS_DOWN:
printf("%s:%s
",enum_btn_id_string[btn->id],enum_event_string[btn->event]);
break;
caseFLEX_BTN_PRESS_CLICK:
HAL_GPIO_WritePin(LED_GPIO_Port,LED_Pin,GPIO_PIN_RESET);//灭
printf("%s:%s
",enum_btn_id_string[btn->id],enum_event_string[btn->event]);
printf("<<<<<<<<<<<<<<<<<<<<<<<;
caseFLEX_BTN_PRESS_DOUBLE_CLICK:
HAL_GPIO_WritePin(LED_GPIO_Port,LED_Pin,GPIO_PIN_RESET);//灭
printf("%s:%s
",enum_btn_id_string[btn->id],enum_event_string[btn->event]);
printf("<<<<<<<<<<<<<<<<<<<<<<<;
default:
break;
}
break;
}

default:
break;
}
}

主函数中需要调用flex_button_scan进行按键扫描,主函数如:

左右滑动查看全部代码>>>

intmain(void)
{
HAL_Init();
SystemClock_Config();
MX_GPIO_Init();
MX_USART1_Init();
printf("微信公众号:嵌入式大杂烩
");
user_button_init();

while(1)
{
flex_button_scan();
HAL_Delay(20);
}
}

编译、下载运行:

6a6bdba2-d8a3-11ec-ba43-dac502259ad0.png6ace98fa-d8a3-11ec-ba43-dac502259ad0.png6af02b6e-d8a3-11ec-ba43-dac502259ad0.png

其中,每次按键的按下都会触发FLEX_BTN_PRESS_DOWN事件。

在对按键进行注册时有设置短按、长按、长按保持的tick:

左右滑动查看全部代码>>>

user_button[i].short_press_start_tick=FLEX_MS_TO_SCAN_CNT(1500);//设置短按事件触发的起始tick
user_button[i].long_press_start_tick=FLEX_MS_TO_SCAN_CNT(3000);//设置长按事件触发的起始tick
user_button[i].long_hold_start_tick=FLEX_MS_TO_SCAN_CNT(4500);//设置长按保持事件触发的起始tick

我们应用比较敏感的是1500、3000、4500,这对应的就是按键按下的时间(单位是ms),即:

  • 按键保持1500ms按下状态时,flexible_button会向应用上报FLEX_BTN_PRESS_SHORT_START事件。
  • 按键保持3000ms按下状态时,flexible_button会向应用上报FLEX_BTN_PRESS_LONG_START事件。
  • 按键保45000ms按下状态时,flexible_button会向应用上报FLEX_BTN_PRESS_LONG_HOLD事件。

下面我们修改代码,按键事件回调函数加入所有事件的处理,触发则打印相应信息。

我们做一个实验,按住button0超过5m,再放开。则打印的信息如:

6b12705c-d8a3-11ec-ba43-dac502259ad0.png

2、基于RT-Thread

FlexibleButton已经有作为一个软件包贡献到RT-Thread中,我们只需要简单的配置,就可以使用了。

RT-Thread menuconfig 方式:

RT-Threadonlinepackages--->
miscellaneouspackages--->
[*]FlexibleButton:Smallandflexiblebuttondriver--->
[*]Enableflexiblebuttondemo
version(latest)--->
6b43d98a-d8a3-11ec-ba43-dac502259ad0.png

配置完成后,输入pkgs --update下载软件包:

6b98b75c-d8a3-11ec-ba43-dac502259ad0.png

运行测试:

6be9877c-d8a3-11ec-ba43-dac502259ad0.png


以上就是本次的分享,希望大家喜欢!文章如有错误,欢迎指出!

审核编辑 :李倩


声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 算法
    +关注

    关注

    23

    文章

    4456

    浏览量

    90849
  • C语言
    +关注

    关注

    180

    文章

    7532

    浏览量

    129858
  • 源码
    +关注

    关注

    8

    文章

    574

    浏览量

    28615
  • RT-Thread
    +关注

    关注

    31

    文章

    1155

    浏览量

    38949

原文标题:超实用!按键原理及应用(附开源代码)

文章出处:【微信号:mcu168,微信公众号:硬件攻城狮】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    4月25日北京站RT-Thread线下workshop,探索RT-Thread混合部署新模式

    4月25日,下午我们将在北京举办RT-Thread混合部署线下workshop,在瑞芯微RK3568平台上实现同时运行RT-Thread和linux,本次workshop邀请到RT-Thread资深
    的头像 发表于 04-16 08:35 106次阅读
    4月25日北京站<b class='flag-5'>RT-Thread</b>线下workshop,探索<b class='flag-5'>RT-Thread</b>混合部署新模式

    4月10日深圳场RT-Thread线下workshop,探索RT-Thread混合部署新模式!

    4月10日我们将在深圳福田举办RT-Thread混合部署线下workshop,在瑞芯微RK3568平台上实现同时运行RT-Thread和linux,本次workshop邀请到RT-Thread资深嵌入式软件工程师农晓明老师为您讲
    的头像 发表于 03-27 11:36 450次阅读
    4月10日深圳场<b class='flag-5'>RT-Thread</b>线下workshop,探索<b class='flag-5'>RT-Thread</b>混合部署新模式!

    4月10日深圳场RT-Thread线下workshop,探索RT-Thread混合部署新模式!

    4月10日我们将在深圳福田举办RT-Thread混合部署线下workshop,在瑞芯微RK3568平台上实现同时运行RT-Thread和linux,本次workshop邀请到RT-Thread资深
    的头像 发表于 03-27 08:34 175次阅读
    4月10日深圳场<b class='flag-5'>RT-Thread</b>线下workshop,探索<b class='flag-5'>RT-Thread</b>混合部署新模式!

    基于RT-Thread os实现混合定位介绍

    wayz_iotkit 是上海图趣信息科技有限公司,针对RT-thread开发的能够实现定位功能的物联网组件。
    的头像 发表于 11-15 16:19 537次阅读
    基于<b class='flag-5'>RT-Thread</b> <b class='flag-5'>os</b>实现混合定位介绍

    RT-thread源码移植到STM32F10x和STM32F4xx

    RT-thread源码移植到STM32F10x和STM32F4xx: 一、源码下载 点击入门->下载   在历史版本里边随便选取一个   会进入百度云盘的下载地址,里边有全部版本的源码
    的头像 发表于 11-15 09:38 1083次阅读
    <b class='flag-5'>RT-thread</b><b class='flag-5'>源码</b>移植到STM32F10x和STM32F4xx

    基于rt-thread的socket通信设计

    最近再研究 rt-thread 的通信 ,想设计出 eps8266(多个) rt-thread(作为中控) 服务器的通信框架,使用的开发板是 潘多拉
    的头像 发表于 10-13 15:02 730次阅读
    基于<b class='flag-5'>rt-thread</b>的socket通信设计

    试用RT-Thread Studio(VSCode)

    想尝试RT-Thread studio (VSCode),先下载安装VSCode,再搜索RT-Thread
    的头像 发表于 10-12 10:58 579次阅读
    试用<b class='flag-5'>RT-Thread</b> Studio(VSCode)

    RT-Thread v5.0.2 发布

    RT-Thread 代码仓库地址: ●  https://github.com/RT-Thread/rt-thread RT-Thread 5.0.2 版本发布日志详情: ●  htt
    的头像 发表于 10-10 18:45 792次阅读
    <b class='flag-5'>RT-Thread</b> v5.0.2 发布

    基于 RT-Thread 的 RoboMaster 电控框架(一)

    由于 RT-Thread 稳定高效的内核,丰富的文档教程,积极活跃的社区氛围,以及设备驱动框架、Kconfig、Scons、日志系统、海量的软件包……很难不选择 RT-Thread 进行项目开发
    的头像 发表于 09-19 19:55 431次阅读

    ESP8266 Non-OS SDK API参考

    电子发烧友网站提供《ESP8266 Non-OS SDK API参考.pdf》资料免费下载
    发表于 09-18 10:08 3次下载
    ESP8266 <b class='flag-5'>Non-OS</b> SDK API参考

    新书上架|RT-Thread带你解密“芯”世界

    夏木荫荫,人间八月,RT-Thread又上新了!自2018年RT-Thread首本书籍问世后,越来越多的创作者因为RT-Thread的自主可控、稳定可靠、友好生态,把RT-Thread
    的头像 发表于 08-22 20:10 462次阅读
    新书上架|<b class='flag-5'>RT-Thread</b>带你解密“芯”世界

    教你手上没有开发板如何跑RT-THREAD STM32应用?

    首先打开 RT-Thread Studio,新建RT-Thread
    的头像 发表于 07-18 16:09 1078次阅读
    教你手上没有开发板如何跑<b class='flag-5'>RT-THREAD</b> STM32应用?

    RT-Thread中的Github Actions

    RT-Thread中一共有五个Github Action(rt-thread/.github/workflow) 分别是: RT-Thread BSP build check(.github
    的头像 发表于 06-01 03:10 360次阅读
    <b class='flag-5'>RT-Thread</b>中的Github Actions

    有CAN驱动程序支持RT-Thread OS吗?

    NXP 有 CAN 驱动程序支持 RT-Thread OS 吗?
    发表于 05-18 08:19

    基于RT-Thread Studio学习

    前期准备:从官网下载 RT-Thread Studio,弄个账号登陆,开启rt-thread学习之旅。
    的头像 发表于 05-15 11:00 2638次阅读
    基于<b class='flag-5'>RT-Thread</b> Studio学习