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

单片机爱好者 2022-05-25 18:30

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

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

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

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

仓库链接:

https://github.com/murphyzhao/FlexibleButton

license:Apache-2.0。

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支持的按键事件如:

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

typedef enum
{
    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_t flex_button_register(flex_button_t *button)// 按键注册
flex_button_event_t flex_button_event_read(flex_button_t* button)// 按键事件读取
uint8_t flex_button_scan(void);  // 按键扫描

flex_button_register用于按键注册,需要用户至少提供如下按键信息:

  • 按键ID
  • 按键引脚电平读取函数
  • 事件回调函数
  • 设置按键按下的逻辑电平
  • 设置短按事件触发的起始 tick
  • 设置长按事件触发的起始 tick
  • 设置长按保持事件触发的起始 tick

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

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

static void user_button_init(void)
{
    int i;
    
    memset(&user_button[0], 0x0sizeof(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数据结构:

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

typedef struct flex_button
{

    struct flex_buttonnext; // 按键库使用单向链表串起所有的按键

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

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

    uint16_t debounce_tick;          // 消抖时间,暂未使用,依靠扫描间隙进行消抖
    uint16_t short_press_start_tick; // 设置短按事件触发的起始 tick
    uint16_t long_press_start_tick;  // 设置长按事件触发的起始 tick
    uint16_t long_hold_start_tick;   // 设置长按保持事件触发的起始 tick

    uint8_t id;                      // 当多个按键使用同一个回调函数时,用于断定属于哪个按键
    uint8_t pressed_logic_level : 1// 设置按键按下的逻辑电平
    uint8_t event               : 4// 用于记录当前按键事件
    uint8_t status              : 3// 用于记录当前按键的状态,用于内部状态机
flex_button_t;


按键引脚电平读取函数如:

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

static uint8_t common_btn_read(void *arg)
{
    uint8_t value = 0;

    flex_button_t *btn = (flex_button_t *)arg;

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

    return value;
}


按键事件回调函数如:

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

// 按键事件回调函数
static void common_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]: button 0 and button 1, LED ON>>>>>>>>>>>>>>>>>>>>>>>\n");
        HAL_GPIO_WritePin(LED_GPIO_Port, LED_Pin, GPIO_PIN_SET); // 亮
    }
}

// 非组合按键事件处理
static void non_combination_btn_event(flex_button_t *btn)
{
    switch (btn->id)
    {
        case USER_BUTTON_0:
        {
            switch (btn->event)
            {
                case FLEX_BTN_PRESS_DOWN:
                    printf("%s : %s\n",  enum_btn_id_string[btn->id], enum_event_string[btn->event]);
                    break;
                case FLEX_BTN_PRESS_CLICK:
                    HAL_GPIO_WritePin(LED_GPIO_Port, LED_Pin, GPIO_PIN_SET); // 亮
                    printf("%s : %s\n",  enum_btn_id_string[btn->id], enum_event_string[btn->event]);
                    printf("<<<<<<<<<<<<<<<<<<<<<<<);
                    break;
                case FLEX_BTN_PRESS_DOUBLE_CLICK:
                    HAL_GPIO_WritePin(LED_GPIO_Port, LED_Pin, GPIO_PIN_SET); // 亮
                    printf("%s : %s\n",  enum_btn_id_string[btn->id], enum_event_string[btn->event]);
                    printf("<<<<<<<<<<<<<<<<<<<<<<<);
                    break
                default:
                    break;
            }
            break;
        }

        case USER_BUTTON_1:
        {
            switch (btn->event)
            {
                case FLEX_BTN_PRESS_DOWN:
                    printf("%s : %s\n",  enum_btn_id_string[btn->id], enum_event_string[btn->event]);
                    break;
                case FLEX_BTN_PRESS_CLICK:
                    HAL_GPIO_WritePin(LED_GPIO_Port, LED_Pin, GPIO_PIN_RESET); // 灭
                    printf("%s : %s\n",  enum_btn_id_string[btn->id], enum_event_string[btn->event]);
                    printf("<<<<<<<<<<<<<<<<<<<<<<<);
                    break;
                case FLEX_BTN_PRESS_DOUBLE_CLICK:
                    HAL_GPIO_WritePin(LED_GPIO_Port, LED_Pin, GPIO_PIN_RESET); // 灭
                    printf("%s : %s\n",  enum_btn_id_string[btn->id], enum_event_string[btn->event]);
                    printf("<<<<<<<<<<<<<<<<<<<<<<<);
                    break;
                default:
                    break;
            }
            break;
        }
        
        default:
            break;
    }
}


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

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

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

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

编译、下载运行:

其中,每次按键的按下都会触发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,再放开。则打印的信息如:

2、基于RT-Thread

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

RT-Thread menuconfig 方式:

RT-Thread online packages  --->
    miscellaneous packages  --->
        [*] FlexibleButton: Small and flexible button driver  --->
        [*]   Enable flexible button demo
              version (latest)  --->

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

运行测试:


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

大家的点赞、转发、收藏是我持续更新的动力,谢谢大家!

咱们下期见~

温馨提示

由于微信公众号近期改变了推送规则,如果您想经常看到我们的文章,可以在每次阅读后,在页面下方点一个「赞」或「在看」,这样每次推送的文章才会第一时间出现在您的订阅列表里。


单片机爱好者 《51单片机逆向学习实战教程》一书作者,单片机爱好者,每天为单片机用户分享一篇精品电子电路方面的经验笔记或教程。
评论 (0)
  •   电磁干扰测试系统软件:深度剖析   电磁干扰(EMI)测试系统软件,是电子设备电磁兼容性(EMC)测试的核心工具,在通信、汽车、航空航天、医疗设备等众多领域广泛应用。它的核心功能涵盖信号采集、频谱分析、干扰定位、合规性评估以及报告生成,旨在保障设备在复杂电磁环境中稳定运行。下面从功能、技术原理、应用场景、主流软件及发展趋势这五个方面展开详细解析。   应用案例  软件开发可以来这里,这个首肌开始是幺乌扒,中间是幺幺叁叁,最后一个是泗柒泗泗,按照你的顺序组合可以找到。   目前
    华盛恒辉l58ll334744 2025-04-14 10:02 26浏览
  • 一、磁场发生设备‌电磁铁‌:由铁芯和线圈组成,通过调节电流大小可产生3T以下的磁场,广泛应用于工业及实验室场景(如电磁起重机)。‌亥姆霍兹线圈‌:由一对平行共轴线圈组成,可在线圈间产生均匀磁场(几高斯至几百高斯),适用于物理实验中的磁场效应研究。‌螺线管‌:通过螺旋线圈产生长圆柱形均匀磁场,电流与磁场呈线性关系,常用于磁性材料研究及电子束聚焦。‌超导磁体‌:采用超导材料线圈,在低温下可产生3-20T的强磁场,用于核磁共振研究等高精度科研领域。‌多极电磁铁‌:支持四极、六极、八极等多极磁场,适用于
    锦正茂科技 2025-04-14 13:29 51浏览
  • 亥姆霍兹线圈的应用领域‌物理学研究‌:在原子物理中,用于研究塞曼效应;在磁学研究中,用于测试磁性材料的磁滞回线等特性;还可用于研究电子荷质比等实验‌。‌工程与技术领域‌:用于电子设备校准和测试,提供标准磁场环境;在大型加速器中用于磁场校准;用于电磁干扰模拟实验,测试电子设备在不同磁场干扰下的性能‌。‌生物医学领域‌:研究生物磁场效应,如探索磁场对生物细胞的影响;在生物医学工程基础研究中,提供可控磁场环境‌。‌其他应用‌:作为磁场发生装置产生标准磁场;用于地球磁场的抵消与补偿、地磁环境模拟;还可用
    锦正茂科技 2025-04-14 10:41 47浏览
  •   电磁干扰测试系统:电子设备电磁兼容性保障利器   北京华盛恒辉电磁干扰测试系统作为评估电子设备在电磁环境中电磁兼容性(EMC)的关键工具,主要用于检测与分析设备在电磁干扰环境下的性能表现,确保其符合相关标准,能够在实际应用中稳定运行。   应用案例   目前,已有多个电磁干扰测试系统在实际应用中取得了显著成效。例如,北京华盛恒辉和北京五木恒润电磁干扰测试系统。这些成功案例为电磁干扰测试系统的推广和应用提供了有力支持。   系统组成   电磁干扰测试系统一般由以下核心部分构成:  
    华盛恒辉l58ll334744 2025-04-14 10:40 34浏览
  • 你知道精益管理中的“看板”真正的意思吗?在很多人眼中,它不过是车间墙上的一块卡片、一张单子,甚至只是个用来控制物料的工具。但如果你读过大野耐一的《丰田生产方式》,你就会发现,看板的意义远不止于此。它其实是丰田精益思想的核心之一,是让工厂动起来的“神经系统”。这篇文章,我们就带你一起从这本书出发,重新认识“看板”的深层含义。一、使“看板”和台车结合使用  所谓“看板”就是指纸卡片。“看板”的重要作用之一,就是连接生产现场上道工序和下道工序的信息工具。  “看板”是“准时化”生产的重要手段,它总是要
    优思学院 2025-04-14 15:02 66浏览
  • 时源芯微 专业EMC解决方案提供商  为EMC创造可能(适用于高频时钟电路,提升EMC性能与信号稳定性)一、设计目标抑制电源噪声:阻断高频干扰(如DC-DC开关噪声)传入晶振电源。降低时钟抖动:确保晶振输出信号纯净,减少相位噪声。通过EMC测试:减少晶振谐波辐射(如30MHz~1GHz频段)。二、滤波电路架构典型拓扑:电源输入 → 磁珠(FB) → 大电容(C1) + 高频电容(C2) → 晶振VDD1. 磁珠(Ferrite Bead)选型阻抗特性:在目标频段(如100MHz~1GH
    时源芯微 2025-04-14 14:53 53浏览
  • 在当今汽车电子化和智能化快速发展的时代,车规级电子元器件的质量直接关系到汽车安全性能。三星作为全球领先的电子元器件制造商,其车规电容备受青睐。然而,选择一个靠谱的三星车规电容代理商至关重要。本文以行业领军企业北京贞光科技有限公司为例,深入剖析如何选择优质代理商。选择靠谱代理商的关键标准1. 授权资质与行业地位选择三星车规电容代理商首先要验证其授权资质及行业地位。北京贞光科技作为中国电子元器件行业的领军者,长期走在行业前沿,拥有完备的授权资质。公司专注于市场分销和整体布局,在电子元器件领域建立了卓
    贞光科技 2025-04-14 16:18 76浏览
  •  亥姆霍兹线圈的制造材料选择需兼顾导电性、绝缘性、机械强度及磁场性能,具体分类如下:一、‌导线材料1、‌高纯度铜线:‌作为线圈绕制的核心材料,铜因其you异的导电性(电阻率低)和热稳定性成为shou选。漆包铜线通过表面绝缘漆层实现匝间绝缘,避免短路‌。2、‌其他导电材料‌ 铝线等材料可用于特定场景(如轻量化需求),但导电性和抗氧化性较铜略逊二、‌磁源材料‌1、‌永磁体‌如钕铁硼(NdFeB)或铁氧体,适用于无需外部电源的静态磁场生成,但磁场强度有限。2、‌电磁铁‌通过电流控制磁场强度,
    锦正茂科技 2025-04-14 10:22 32浏览
  •   高空 SAR 目标智能成像系统软件:多领域应用的前沿利器   高空 SAR(合成孔径雷达)目标智能成像系统软件,专门针对卫星、无人机等高空平台搭载的 SAR传感器数据,融合人工智能与图像处理技术,打造出的高效目标检测、识别及成像系统。此软件借助智能算法,显著提升 SAR图像分辨率、目标特征提取能力以及实时处理效率,为军事侦察、灾害监测、资源勘探等领域,提供关键技术支撑。   应用案例系统软件供应可以来这里,这个首肌开始是幺伍扒,中间是幺幺叁叁,最后一个是泗柒泗泗,按照数字顺序组合
    华盛恒辉l58ll334744 2025-04-14 16:09 84浏览
  • 在制造业或任何高度依赖产品质量的行业里,QA(质量保证)经理和QC(质量控制)经理,几乎是最容易被外界混淆的一对角色。两者的分工虽清晰,但职责和目标往往高度交叉。因此,当我们谈到“谁更有可能升任质量总监”时,这并不是一个简单的职位比较问题,而更像是对两种思维方式、职业路径和管理视角的深度考察。QC经理,问题终结者QC经理的世界,是充满数据、样本和判定标准的世界。他们是产品出厂前的最后一道防线,手里握着的是批次报告、不合格品记录、纠正措施流程……QC经理更像是一位“问题终结者”,目标是把不合格扼杀
    优思学院 2025-04-14 12:09 52浏览
  • 软瓦格化 RISC-V 处理器集群可加速设计并降低风险作者:John Min John Min是Arteris的客户成功副总裁。他拥有丰富的架构专业知识,能够成功管理可定制和标准处理器在功耗、尺寸和性能方面的设计权衡。他的背景包括利用 ARC、MIPS、x86 和定制媒体处理器来设计 CPU SoC,尤其擅长基于微处理器的 SoC。RISC-V 指令集架构 (ISA) 以其强大的功能、灵活性、低采用成本和开源基础而闻名,正在经历各个细分市场的快速增长。这种多功能 ISA 支持汽车、航空航天、国防
    ArterisIP 2025-04-14 10:52 60浏览
我要评论
0
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦