本文来源电子发烧友社区,作者:juby, 帖子地址:https://bbs.elecfans.com/jishu_2010914_1_1.html
长按、短按的应用

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);
}
初始化定时器
if (ret != HI_ERR_SUCCESS)
{
printf("timer create failrn");
}
printf("timer create successrn");
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");
}
* 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);
定时器回调函数
{
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);
}
}
结果展示

资料获取
长按、短按的应用
我们之前在下面网文中介绍过了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);
}
}
结果展示

资料获取
公众号留言区置顶留言获取本文对应示例源码。
ps: 文章首发于电子发烧友。
欢迎关注
程序员小哈带你玩转嵌入式,微信搜索:嵌入式从0到1,更多干货等着你。
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。
举报投诉
-
wi-fi
+关注
关注
15文章
2458浏览量
129992 -
HarmonyOS
+关注
关注
80文章
2156浏览量
36277 -
HiSpark
+关注
关注
1文章
156浏览量
7797
发布评论请先 登录
相关推荐
热点推荐
技术资讯 I Wi-Fi 模块设计
本文要点Wi-Fi模块设计旨在打造一套紧凑且高效的硬件和软件解决方案,使设备能够通过Wi-Fi网络实现通信。Wi-Fi模块设计是一个复杂的过程,要求研发人员兼具硬件与软件工程方面的专业
ESP32 Wi-Fi 控制 LED 灯的原理
在智能家居、物联网设备中,用手机通过Wi-Fi控制灯光、风扇或插座,已经非常普遍。而在嵌入式开发中,ESP32是最常用的Wi-Fi模块之一。本文将带你系统理解:ESP32是如何通过Wi-Fi
Wi-Fi:无线连接的全球通用语
一、什么是Wi-Fi?Wi-Fi是Wi-Fi联盟制造商的商标认证,是基于IEEE 802.11标准的无线局域网技术。它允许电子设备在特定范围内无线接入网络,实现高速数据交换与互联网访问
发表于 01-07 09:49
Nordic发布nRF7002 EBII 开发板, 支持Wi-Fi 6, 解锁nRF54L新玩法
和 5 GHz)以及高级 Wi-Fi 6 功能,例如目标唤醒时间 (TWT)、OFDMA 和 BSS 着色,从而实现高效、无干扰的电池供电运行。它采用双频芯片天线,确保在各个 Wi-Fi 频段上都能提供
发表于 12-10 11:58
Wi-Fi模块选型指南 | 低功耗蓝牙/Wi-Fi 6模块推荐 | 物联网设备集成攻略
物联网的快速发展,Wi-Fi技术的成熟的情况下,越来越多的场景需要用到Wi-Fi来无线传输数据,尤其是移动设备中,怎么再设备中选择合适的Wi-Fi模块呢?下面我们来介绍下Wi-Fi模块
浅谈Wi-Fi 6E与Wi-Fi 7的关键器件——BAW滤波器新技术
作者: Qorvo 亚太区无线连接事业部高级行销经理林健富 2020年1月,Wi-Fi联盟正式宣布开放6GHz频段(5925MHz-7125MHz),并将其命名为Wi-Fi 6E。2020年4月
发表于 09-19 18:29
•2437次阅读
解读Nordic基于SSID的Wi-Fi定位解决方案
,与 nRF91 系列蜂窝物联网模组配合使用,可实现基于 SSID 的 Wi-Fi 定位。Nordic基于SSID的Wi-Fi定位可以在室内和室外、城市和郊区以极其省电的方式获取精确的位置信息。这是对全球导航卫星
基于 SSID 的 Wi-Fi 定位:与其他定位服务的性能比较
,与 nRF91 系列蜂窝物联网模组配合使用,可实现基于 SSID 的 Wi-Fi 定位。Nordic基于SSID的Wi-Fi定位可以在室内和室外、城市和郊区以极其省电的方式获取精确的位置信息。这是对全球
发表于 08-31 21:01
LitePoint Wi-Fi测试软件减轻客户设计负担
自Wi-Fi 7于一年多前获得Wi-Fi联盟认证以来,作为最新一代通信技术,Wi-Fi正逐步成为用户实现无所不在无线连接的新选择。随着每一代Wi-F
有没有什么修复方法可以确保 AP 模式下的 Wi-Fi 和 BLE 连接同时正常工作?
我正在使用 CYBSYSKIT DEV 01 套件。我尝试在 AP 模式下打开 Wi-Fi 并宣传 BLE。我可以宣传 SoftAP 和 BLE。但是,我无法从中央设备连接到 BLE。它可以立即连接并断开连接。有没有什么修复方法可以确保 AP 模式下的
发表于 07-17 06:13
【HarmonyOS 5】金融应用开发鸿蒙组件实践
原生鸿蒙操作系统星河版,面向开发者开放申请,余承东宣布鸿蒙生态设备数达 8 亿台;建设银行、邮储银行等完成鸿蒙原生应用 Beta 版本开发。 2024 年 10 月 22 日:
Wi-Fi 8:开启极高可靠性 (UHR) 连接的新纪元——2
Aggregation)。由于所有操作均可跨频段与频道进行,因此显著提升了整个网络系统的数据传输速度,并有效降低了多用户同时在线传输时所产生的延迟问题。图3展示了Wi-Fi 7中 MLO技术如何通过不同频段
发表于 06-13 11:15
Wi-Fi 8:开启极高可靠性 (UHR) 连接的新纪元——1
就开始启动Wi-Fi 8的项目研究?原因在于,Wi-Fi 7的技术已经正式落地,芯片与解决方案的提供商以及电信服务的运营商也已开始布局并逐渐完善Wi-Fi 7的生态系统。厂商们希望借此
发表于 06-13 11:09
Wi-Fi HaLow如何突破传统Wi-Fi性能瓶颈?
视频推荐在万物互联的时代,传统Wi-Fi在覆盖范围、功耗、连接数上渐显不足。而Wi-FiHaLow的诞生,正在用黑科技打破传统Wi-Fi的性能瓶颈!今天,我们一起来探究Wi-FiHaL
【HarmonyOS HiSpark Wi-Fi IoT 套件试用连载】基于鸿蒙操作系统的单个按键长按、短按的实现
评论