ESP-IDF使用Button组件实现按键检测的功能

2024-05-24 12:12

本文主要是介绍ESP-IDF使用Button组件实现按键检测的功能,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

ESP32使用Button组件实现按键检测的功能

    • ESP-IDF 组件管理
    • LED 组件简介
    • 测试button组件
    • 写在最后

ESP-IDF 组件管理

IDF 组件管理器工具用于下载 ESP-IDF CMake 项目的依赖项,该下载在 CMake 运行期间自动完成。IDF 组件管理器可以从自动从组件注册表 或 Git 仓库获取组件,具体的使用和介绍,请查看官方文档,需要注意的是 ESP-IDF 管理器是 IDF4.4 以后引入的特性,使用IDF框架时,建议尽量使用最新的 release 版本 IDF 构建自己的项目。

要获取组件列表,请参阅 https://components.espressif.com/

LED 组件简介

  1. 我们从 ESP-IDF 组件管理器网页搜索我们需要的组件【button】,然后 点开 espressif/button 组件
    在这里插入图片描述
    在这里插入图片描述

  2. button 组件简介

Button按键组件实现了 GPIO 和 ADC 两种按键,并允许同时创建两种不同的按键。下图显示了两种按键的硬件设计:
在这里插入图片描述

  • GPIO 按键优点有:每一个按键占用独立的 IO,之间互不影响,稳定性高;缺点有:按键数量多时占用太多 IO 资源。

  • ADC 按键优点有:可多个按键共用一个 ADC 通道,占用 IO 资源少;缺点有:不能同时按下多按键,当按键因氧化等因素导致闭合电阻增大时,容易误触,稳定性不高。
    每个按键拥有下表的 8 个事件:
    在这里插入图片描述
    每个按键可以有 回调轮询 两种使用方式:

  • 回调:一个按键的每个事件都可以为其注册一个回调函数,产生事件时回调函数将会被调用。这种方式的效率和实时性高,不会丢失事件。

  • 轮询:在程序中周期性调用 :ciot_button_get_event 查询按键当前的事件。
    这种方式使用简单,适合任务简单的场合,当然你也可以将以上两种方式组合使用。

注意:回调函数中不能有 TaskDelay 等阻塞的操作

具体功能和使用方法,请查看官方文档

测试button组件

  1. 使用 VSCODE 的 ESP-IDF 插件创建一个 helloword 的工程
    在这里插入图片描述
  2. 复制button组件界面上配置组件的命令 idf.py add-dependency "espressif/button^3.2.0" 在当前 helloworld 工程下增加 button组件
    在这里插入图片描述

PS:遇到的一些问题及解决方法:

  • idf.py 的命令无法执行,提示 idf.py : 无法将 “idf.py ”项识别为 cmet、函数、脚本文件或可运行程序的名称。请检査名称的拼写,如果包括路径,请确保路径正确,然后再试一次。 或者只会打开 idf.py 这个文件而不执行这个文件的内容.
    解决方法:
    指令只能在 ESP-IDF Terminal 中执行,不能在系统自带的 powershell 等工具中执行
    在这里插入图片描述
  1. 生成 idf_component.yml 文件后,使用命令 idf.py build 或者插件上面的编译按钮对整个工程先进行一次编译,编译完成后,组件就会出现在 managed_components 目录下
    在这里插入图片描述

  2. 此处我们测试最简单GPIO控制LED的功能;将D:\Espressif\MyProject\hello_world\managed_components\espressif__button\examples\button_power_save\main\main.c 文件里面的内容全部复制到 hello_world_main.c
    在这里插入图片描述

并对 hello_world_main.c 的内容进行稍微的修改,因为我们不测试低功耗的相关功能,故删除低功耗测试的相关代码,并且修改 GPIO 及按键按下的有效电平;以下使用IO0作为按键输入,且有效电平为低电平。

/** SPDX-FileCopyrightText: 2023-2024 Espressif Systems (Shanghai) CO LTD** SPDX-License-Identifier: Apache-2.0*/#include <stdio.h>
#include "freertos/FreeRTOS.h"
#include "freertos/task.h"
#include "esp_log.h"
#include "esp_pm.h"
#include "iot_button.h"
#include "esp_sleep.h"
#include "esp_idf_version.h"/* Most development boards have "boot" button attached to GPIO0.* You can also change this to another pin.*/
#if CONFIG_IDF_TARGET_ESP32C3 || CONFIG_IDF_TARGET_ESP32C2 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32C6
#define BOOT_BUTTON_NUM         9
#else
#define BOOT_BUTTON_NUM         0
#endif
#define BUTTON_ACTIVE_LEVEL     0static const char *TAG = "button_test";const char *button_event_table[] = {"BUTTON_PRESS_DOWN","BUTTON_PRESS_UP","BUTTON_PRESS_REPEAT","BUTTON_PRESS_REPEAT_DONE","BUTTON_SINGLE_CLICK","BUTTON_DOUBLE_CLICK","BUTTON_MULTIPLE_CLICK","BUTTON_LONG_PRESS_START","BUTTON_LONG_PRESS_HOLD","BUTTON_LONG_PRESS_UP",
};static void button_event_cb(void *arg, void *data)
{ESP_LOGI(TAG, "Button event %s", button_event_table[(button_event_t)data]);#if CONFIG_GPIO_BUTTON_SUPPORT_POWER_SAVEesp_sleep_wakeup_cause_t cause = esp_sleep_get_wakeup_cause();if (cause != ESP_SLEEP_WAKEUP_UNDEFINED) {ESP_LOGI(TAG, "Wake up from light sleep, reason %d", cause);}#endif
}void button_init(uint32_t button_num)
{button_config_t btn_cfg = {.type = BUTTON_TYPE_GPIO,.gpio_button_config = {.gpio_num = button_num,.active_level = BUTTON_ACTIVE_LEVEL,
#if CONFIG_GPIO_BUTTON_SUPPORT_POWER_SAVE.enable_power_save = true,
#endif},};button_handle_t btn = iot_button_create(&btn_cfg);assert(btn);esp_err_t err = iot_button_register_cb(btn, BUTTON_PRESS_DOWN, button_event_cb, (void *)BUTTON_PRESS_DOWN);err |= iot_button_register_cb(btn, BUTTON_PRESS_UP, button_event_cb, (void *)BUTTON_PRESS_UP);err |= iot_button_register_cb(btn, BUTTON_PRESS_REPEAT, button_event_cb, (void *)BUTTON_PRESS_REPEAT);err |= iot_button_register_cb(btn, BUTTON_PRESS_REPEAT_DONE, button_event_cb, (void *)BUTTON_PRESS_REPEAT_DONE);err |= iot_button_register_cb(btn, BUTTON_SINGLE_CLICK, button_event_cb, (void *)BUTTON_SINGLE_CLICK);err |= iot_button_register_cb(btn, BUTTON_DOUBLE_CLICK, button_event_cb, (void *)BUTTON_DOUBLE_CLICK);err |= iot_button_register_cb(btn, BUTTON_LONG_PRESS_START, button_event_cb, (void *)BUTTON_LONG_PRESS_START);err |= iot_button_register_cb(btn, BUTTON_LONG_PRESS_HOLD, button_event_cb, (void *)BUTTON_LONG_PRESS_HOLD);err |= iot_button_register_cb(btn, BUTTON_LONG_PRESS_UP, button_event_cb, (void *)BUTTON_LONG_PRESS_UP);ESP_ERROR_CHECK(err);
}
#if CONFIG_GPIO_BUTTON_SUPPORT_POWER_SAVE
#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 1, 0)
void power_save_init(void)
{esp_pm_config_t pm_config = {.max_freq_mhz = CONFIG_EXAMPLE_MAX_CPU_FREQ_MHZ,.min_freq_mhz = CONFIG_EXAMPLE_MIN_CPU_FREQ_MHZ,
#if CONFIG_FREERTOS_USE_TICKLESS_IDLE.light_sleep_enable = true
#endif};ESP_ERROR_CHECK(esp_pm_configure(&pm_config));
}
#else
void power_save_init(void)
{
#if CONFIG_IDF_TARGET_ESP32esp_pm_config_esp32_t pm_config = {
#elif CONFIG_IDF_TARGET_ESP32S2esp_pm_config_esp32s2_t pm_config = {
#elif CONFIG_IDF_TARGET_ESP32C3esp_pm_config_esp32c3_t pm_config = {
#elif CONFIG_IDF_TARGET_ESP32S3esp_pm_config_esp32s3_t pm_config = {
#elif CONFIG_IDF_TARGET_ESP32C2esp_pm_config_esp32c2_t pm_config = {
#endif.max_freq_mhz = CONFIG_EXAMPLE_MAX_CPU_FREQ_MHZ,.min_freq_mhz = CONFIG_EXAMPLE_MIN_CPU_FREQ_MHZ,
#if CONFIG_FREERTOS_USE_TICKLESS_IDLE.light_sleep_enable = true
#endif};ESP_ERROR_CHECK(esp_pm_configure(&pm_config));
}
#endif
#endifvoid app_main(void)
{#if CONFIG_GPIO_BUTTON_SUPPORT_POWER_SAVEpower_save_init();
#endifbutton_init(BOOT_BUTTON_NUM);
}
  1. 对修改完成的功能进行编译,并下载到测试的开发板上,按下接到 IO0 上的按键,就可以通过日志参看按键的事件:
    在这里插入图片描述

写在最后

如果觉得文章对您有帮助,请关注、点赞、收藏,谢谢!

这篇关于ESP-IDF使用Button组件实现按键检测的功能的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

sky-take-out项目中Redis的使用示例详解

《sky-take-out项目中Redis的使用示例详解》SpringCache是Spring的缓存抽象层,通过注解简化缓存管理,支持Redis等提供者,适用于方法结果缓存、更新和删除操作,但无法实现... 目录Spring Cache主要特性核心注解1.@Cacheable2.@CachePut3.@Ca

C#下Newtonsoft.Json的具体使用

《C#下Newtonsoft.Json的具体使用》Newtonsoft.Json是一个非常流行的C#JSON序列化和反序列化库,它可以方便地将C#对象转换为JSON格式,或者将JSON数据解析为C#对... 目录安装 Newtonsoft.json基本用法1. 序列化 C# 对象为 JSON2. 反序列化

QT Creator配置Kit的实现示例

《QTCreator配置Kit的实现示例》本文主要介绍了使用Qt5.12.12与VS2022时,因MSVC编译器版本不匹配及WindowsSDK缺失导致配置错误的问题解决,感兴趣的可以了解一下... 目录0、背景:qt5.12.12+vs2022一、症状:二、原因:(可以跳过,直奔后面的解决方法)三、解决方

Debian 13升级后网络转发等功能异常怎么办? 并非错误而是管理机制变更

《Debian13升级后网络转发等功能异常怎么办?并非错误而是管理机制变更》很多朋友反馈,更新到Debian13后网络转发等功能异常,这并非BUG而是Debian13Trixie调整... 日前 Debian 13 Trixie 发布后已经有众多网友升级到新版本,只不过升级后发现某些功能存在异常,例如网络转

MySQL中On duplicate key update的实现示例

《MySQL中Onduplicatekeyupdate的实现示例》ONDUPLICATEKEYUPDATE是一种MySQL的语法,它在插入新数据时,如果遇到唯一键冲突,则会执行更新操作,而不是抛... 目录1/ ON DUPLICATE KEY UPDATE的简介2/ ON DUPLICATE KEY UP

Python中Json和其他类型相互转换的实现示例

《Python中Json和其他类型相互转换的实现示例》本文介绍了在Python中使用json模块实现json数据与dict、object之间的高效转换,包括loads(),load(),dumps()... 项目中经常会用到json格式转为object对象、dict字典格式等。在此做个记录,方便后续用到该方

JWT + 拦截器实现无状态登录系统

《JWT+拦截器实现无状态登录系统》JWT(JSONWebToken)提供了一种无状态的解决方案:用户登录后,服务器返回一个Token,后续请求携带该Token即可完成身份验证,无需服务器存储会话... 目录✅ 引言 一、JWT 是什么? 二、技术选型 三、项目结构 四、核心代码实现4.1 添加依赖(pom

SpringBoot路径映射配置的实现步骤

《SpringBoot路径映射配置的实现步骤》本文介绍了如何在SpringBoot项目中配置路径映射,使得除static目录外的资源可被访问,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一... 目录SpringBoot路径映射补:springboot 配置虚拟路径映射 @RequestMapp

RabbitMQ 延时队列插件安装与使用示例详解(基于 Delayed Message Plugin)

《RabbitMQ延时队列插件安装与使用示例详解(基于DelayedMessagePlugin)》本文详解RabbitMQ通过安装rabbitmq_delayed_message_exchan... 目录 一、什么是 RabbitMQ 延时队列? 二、安装前准备✅ RabbitMQ 环境要求 三、安装延时队

Python与MySQL实现数据库实时同步的详细步骤

《Python与MySQL实现数据库实时同步的详细步骤》在日常开发中,数据同步是一项常见的需求,本篇文章将使用Python和MySQL来实现数据库实时同步,我们将围绕数据变更捕获、数据处理和数据写入这... 目录前言摘要概述:数据同步方案1. 基本思路2. mysql Binlog 简介实现步骤与代码示例1