【MultiButton】并发状态事件驱动按键驱动模块,量产产品实践

2019-12-11 18:36发布

本帖最后由 半导体 于 2016-9-2 14:18 编辑

简介:
        MultiButton 是一个小巧简单易用的事件驱动型按键驱动模块,可无限量扩展按键,按键事件的回调处理方式可以简化你的程序逻辑,去除冗余的按键处理硬编码,专注你的程序思路。
该模块已稳定用于公司多款量产智能硬件产品。

使用方法:
        1. 先申请一个按键结构
                struct Button button1;

        2. 初始化按键对象,绑定按键连接的GPIO引脚,read_button_pin() 为按键的GPIO读取函数,后一个参数为设置触发电平
                button_init(&button1, read_button_pin, 0);

        3. 注册按键事件,共有以下5种事件:
                        CLICK                        //每次按下都会触发
                        PRESSED                   //单击触发
                        DOUBLE_CLICK           //双击触发
                        LONG_RRESS_START   //长按开始触发一次
                        LONG_PRESS_HOLD    //长按过程一直触发
                        LONG_PRESS_STOP    //长按松手触发

                button_attach(&button1, SINGLE_CLICK, Callback_SINGLE_CLICK_Handler);
                button_attach(&button1, DOUBLE_CLICK, Callback_DOUBLE_Click_Handler);
                ...

        4. 启动按键
                button_start(&button1);

        5. 设置一个5ms间隔的定时器循环调用后台处理函数
                while(1) {
                        ....
                            if(timer_ticks == 5) {
                                timer_ticks = 0;
                                button_ticks();
                            }
                }


特性:
        MultiButton 使用C语言实现,基于面向对象方式设计思路,每个按键对象单独用一份数据结构管理:
  1. struct Button {
  2.     uint16_t ticks;
  3.     uint8_t  state : 3;
  4.     uint8_t  debounce_cnt : 3;
  5.     uint8_t  active_level : 1;
  6.     uint8_t  button_level : 1;
  7.     uint8_t  (*hal_button_Level)(void);
  8.     CallBackFunc  cb[number_of_event];
  9.     struct Button* next;
  10. };
复制代码
这样每个按键使用单向链表相连,依次进入 button_handler(struct Button* handle) 状态机处理,所以每个按键的状态彼此独立。


Examples
  1. #include "button.h"

  2. struct Button button1;
  3. struct Button button2;

  4. int read_button1_pin()
  5. {
  6.     return HAL_GPIO_ReadPin(B1_GPIO_Port, B1_Pin);  //HAL GPIO read.
  7. }
  8. ...

  9. int main()
  10. {
  11.     button_init(&button1, read_button1_pin, 0);  //初始化,绑定按键GPIO电平读取接口
  12.     button_init(&button2, read_button2_pin, 0);

  13.     button_attach(&button1, PRESSED, BTN1_SINGLE_CLICK_Handler);    //绑定 按键1 的单击事件回调
  14.     button_attach(&button1, DOUBLE_CLICK, BTN1_DOUBLE_Click_Handler);    //双击事件回调
  15.     button_attach(&button2, LONG_RRESS_START, BTN2_LONG_RRESS_START_Handler);  //绑定 按键2 长按开始事件回调
  16.     button_attach(&button2, LONG_PRESS_HOLD,  BTN2_LONG_PRESS_HOLD_Handler);
  17.     button_attach(&button2, LONG_PRESS_STOP,  BTN2_LONG_PRESS_STOP_Handler);

  18.     button_start(&button1);  //启动
  19.     button_start(&button2);
  20.    
  21.     //定时循环调用button_ticks() 后台处理函数,该调用方法由你的平台自行实现。
  22.     __timer_start(button_ticks, 0, 5);  
  23.    
  24.     while(ture)
  25.     {
  26.      ...
  27.     }
  28. }

  29. void BTN1_SINGLE_CLICK_Handler()
  30. {
  31.     //do something..
  32. }

  33. void BTN1_DOUBLE_Click_Handler()
  34. {
  35.     //do something..
  36. }
  37. ...
复制代码

模块下载地址:
https://github.com/0x1abin/MultiButton
友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
该问题目前已经被作者或者管理员关闭, 无法添加新回复
99条回答
fenjinzhe
1楼-- · 2019-12-21 00:32
看起来很强大,顶一下。
半导体
2楼-- · 2019-12-21 01:37
单飞 发表于 2016-9-12 14:59
其实我说的这种情况在键盘上是有的,例如:windows键,单独按下是没响应,弹起时有相应(打开开始菜单) ...

对,那你的按键是定义抬起时才先响应操作的话,那一切都好办了,哈哈。
单飞
3楼-- · 2019-12-21 05:13
 精彩回答 2  元偷偷看……
半导体
4楼-- · 2019-12-21 07:46
单飞 发表于 2016-9-12 15:07
63楼就是站在这个基础上讲的,不过还是没想到好的解决方案。
第二个组合键容易误触,误触之后还会和组合 ...

感觉看似简单的一个按键也是一门大学问啊,它直接关系到用户跟产品的交互体验。
单飞
5楼-- · 2019-12-21 08:13
半导体 发表于 2016-9-12 15:20
感觉看似简单的一个按键也是一门大学问啊,它直接关系到用户跟产品的交互体验。 ...

我觉得这个问题需要按键驱动的支持
单飞
6楼-- · 2019-12-21 09:19
再支持一下楼主。
一般我们(可能只有我),习惯的写法不是把按键操作都汇总到一个回调函数里的,感觉多界面操作的时候太麻烦了,也不方便,而且和驱动层不好分离。
习惯这种写法:
void main(void)
{
    u8 key_val = 0;
   
    while(1)
    {
        key_val = bsp_GetKey();//读按键
        
        if(key_val == KEY_NONE) continue;//按键未按下
        
        switch(key_val)
        {
            case KEY_LEFT_PRESS: //【左】
            {
                printf("左键 ");
            }break;
            
            case KEY_RIGHT_PRESS: //【右】
            {
                printf("右键 ");
            }break;
            
            case KEY_UP_PRESS: //【上】
            {
                printf("上键 ");
            }break;
            
            case KEY_DOWN_UNDO: //【下】
            {
                printf("下键 ");
            }break;
            
            default: break;
        }
    }
}

一周热门 更多>