描述
本文来源电子发烧友社区,作者:juby, 帖子地址:https://bbs.elecfans.com/jishu_2010914_1_1.html
长按、短按的应用
我们之前在下面网文中介绍过了ESP8266模块的配网:
Windows下AliOS Things环境搭建及ESP8266 固件下载
固件使用AliOS Things固件的ESP8266模块进行配网的时候,文中是这么操作的:
使用一个跳线,先把D5(GPIO14)接GND,再接3.3V,出现如下Log即进入配网模式:
这个过程其实就是模拟了一个按键长按过程。
长按、短按的原理
我们学习嵌入式要学习其原理,原理学会了,其他平台下相同功能的实现也就会了。
通过阅读AliOS Things 3.0的源码,其中按键状态判断的过程如下:
源文件: platform/mcu/esp8266/bsp/key.c
上述过程简单描述过程如下:
-
按键对应的GPIO中断函数中,开启定时器;
-
定时器响应函数中,循环判断此GPIO的状态。当按键仍为按下状态时,定时计数+1;如果按键变为了释放状态,则停止定时器,计算按键被按下状态总的持续时间;
-
根据时间长短进而判断出此次按键为长按还是短按,进而可以实现一个按键对应多个不同功能。
这种驱动方式跟下面按键驱动方式有明显的优势:
基于鸿蒙OS的按键驱动
此方法优点:天然去抖动,不用延时等待按键状态改变,程序运行效率大大提高。
鸿蒙系统实现单个按键的长按和短按
参考上面原理,我们实现一个鸿蒙系统下的按键长按和短按判断。
初始化GPIO中断
在入口函数SYS_RUN(KeyExampleEntry);中,将GPIO_5设置为下降沿触发中断:
hi_u32 ret = 0;GpioInit();IoSetFunc(WIFI_IOT_IO_NAME_GPIO_5, WIFI_IOT_IO_FUNC_GPIO_5_GPIO);GpioSetDir(WIFI_IOT_GPIO_IDX_5, WIFI_IOT_GPIO_DIR_IN);//IoSetPull(WIFI_IOT_IO_NAME_GPIO_5,WIFI_IOT_IO_PULL_UP);if (ret != WIFI_IOT_SUCCESS) {printf("===== ERROR ======gpio -> GpioSetDir ret:%drn", ret);return;}ret = GpioRegisterIsrFunc(WIFI_IOT_GPIO_IDX_5,WIFI_IOT_INT_TYPE_EDGE,WIFI_IOT_GPIO_EDGE_FALL_LEVEL_LOW, gpio5_isr_func, NULL);if (ret != WIFI_IOT_SUCCESS) {printf("===== ERROR ======gpio -> hi_gpio_register_isr_function ret:%drn", ret);}初始化定时器
在入口函数SYS_RUN(KeyExampleEntry);中创建定时器:
ret = hi_timer_create(&g_timer_handle); if (ret != HI_ERR_SUCCESS){ printf("timer create failrn"); } printf("timer create successrn");
在GPIO_5的中断处理函数中,使用hi_timer_start()函数开启定时器。
/* gpio callback func */void gpio5_isr_func(char *arg){ (void)arg; //临时取消GPIO_5的中断响应 GpioUnregisterIsrFunc(WIFI_IOT_GPIO_IDX_5); printf("----- gpio05 isr success -----rn"); hi_u32 ret = 0; //启动定时器 ret = hi_timer_start(g_timer_handle, HI_TIMER_TYPE_PERIOD, 10, app_demo_timer_handle, 0); if (ret != HI_ERR_SUCCESS) { printf("timer start failrn"); } printf("timer start successrn");}
定时器开始函数定义如下:
* timer_handle,定时器句柄。* type,定时器类型。* expire,定时器超时时间(单位:ms)。配置为0时,默认为10ms。* timer_func,定时器回调函数。* data,回调函数传参。** 返回值0,代表操作成功,* 其他代表失败, 具体定义详见: hi_errno.h。** 依赖:hi_timer.h:文件用于描述定时器相关接口。* 定时器停止使用 hi_timer_stop() 函数。*/hi_u32 hi_timer_start(hi_u32 timer_handle, hi_timer_type type, hi_u32 expire, hi_timer_callback_f timer_func, hi_u32 data);定时器回调函数
在定时器回调函数中,循环判断GPIO_5的状态,只要按键没有释放,就讲计数器自加,每增加1,代表10ms,当按键释放之后,停止计时,最终根据定时器长度来判断此次按键的长短。
static hi_void app_demo_timer_handle(hi_u32 data){ hi_unref_param(data); hi_u32 ret = 0; //定时器计数+1 nCurrentTimerCount++; //每一秒打印一次日志,方便调试 if((nCurrentTimerCount % 100) == 0) printf("count = %d rn",nCurrentTimerCount); WifiIotGpioValue wigv; //获取GPIO_5的状态 GpioGetInputVal(WIFI_IOT_IO_NAME_GPIO_5,&wigv); if (wigv == WIFI_IOT_GPIO_VALUE0) { //按键尚未释放 } else { //停止定时器 ret = hi_timer_stop(g_timer_handle); if (ret != HI_ERR_SUCCESS) { printf("timer stop failrn"); } else { printf("app demo timer stop , count = %d rn",nCurrentTimerCount); //根据按键持续时间判断此次按键操作为长按还是短按 if (nCurrentTimerCount > 600) { nCurrentTimerCount = 0; printf("long long press key rn"); } else if (nCurrentTimerCount > 200) { nCurrentTimerCount = 0; printf("long press key rn"); } else if (nCurrentTimerCount > 4) { nCurrentTimerCount = 0; printf("short press key rn"); } } //恢复GPIO_5的中断响应 ret = GpioRegisterIsrFunc(WIFI_IOT_GPIO_IDX_5,WIFI_IOT_INT_TYPE_EDGE,WIFI_IOT_GPIO_EDGE_FALL_LEVEL_LOW, gpio5_isr_func, NULL); } }结果展示资料获取
公众号留言区置顶留言获取本文对应示例源码。
欢迎关注
程序员小哈带你玩转嵌入式,微信搜索:嵌入式从0到1,更多干货等着你。
打开APP阅读更多精彩内容