开源按键组件MultiButton支持菜单操作(事件驱动型)

简介: 开源按键组件MultiButton支持菜单操作(事件驱动型)

之前一个老友写的MultiButton开源按键组件的剖析讲解,它的设计思想简洁且高效,这篇文章我上周也分享出来给大家共同来学习了。


第1期 | MultiButton,一个小巧简单易用的事件驱动型按键驱动模块


至于介绍和使用在这里我就不多说了,相信看上面这篇文章你应该就懂了,但我想,能不能跟菜单操作绑定在一块呢?这样我不就可以利用起来,实现一个高效稳定的菜单+按键结合的状态机框架?纵观网上很多写菜单框架的,要不写得太死板,要不写得太冗长了,还有的写的还很难看得懂,超级麻烦,虽然各有各的方法去实现,都对,但有些真的不好维护和升级,比如下面这个,这应该是网上广泛流传的一个菜单框架的版本。但我觉得我实在看不下去了。

640.png

640.png

640.png

640.png

还有一些年代久远的论文:

https://wenku.baidu.com/view/576bae38ee06eff9aef807b9.html
https://blog.csdn.net/embedded_guzi/article/details/35835755
https://blog.csdn.net/cjqqschoolqq/article/details/8701387

我的项目设计原则:简单看得懂,实用,稳定,高效可拓展。


通常一些手持式设备都会有各种各样的按键,比如左、中、右、确定、返回、电源键等等(非矩阵键盘),以我目前公司的产品,一般就这几个按键。


我们公司的产品主要的业务逻辑还是应用,应用逻辑最大的工作量其实就是利用按键+页面的形式来体现,这就少不了需要实现一套简单、高效、稳定、可拓展、可维护的菜单+按键的软件框架,有了这么一套好用的框架,后面随便换一个产品,就不用重新再去开发了,我们就可以专注于应用实现,把精力放在更有意义的软件业务逻辑上来。接下来我们在MultiButton的.h文件中添加菜单框架相关的结构体以及一些枚举:

/*菜单,具体是哪个页面,这个留给用户自己去添加*/
typedef enum
{
    MAIN_PAGE = 0,
    LOG_PAGE,
} MENU;
/*事件值,可以留给用户自己去定义,这里我定义了一些我需要的事件*/
typedef enum
{
    /*第一个事件为-1用来定义一个防止重复触发的值*/
    NULL_KEY_EVENT = -1,
    LEFT_KEY_SHORT = 0,
    LEFT_KEY_LONG  = 1,
    ENTER_KEY_SHORT = 2,
    ENTER_KEY_LONG = 3,
    RIGHT_LEY_SHORT = 4,
    RIGHT_KEY_LONG = 5,
    UP_KEY_SHORT = 6,
    UP_KEY_LONG = 7,
    DOWN_KEY_SHORT = 6,
    DOWN_KEY_LONG = 7,
    RETRUN_KEY_SHORT = 8,
    RETRUN_KEY_LONG = 9,
    POWER_KEY_SHORT = 10,
    POWER_KEY_LONG = 11,
} EVENT_CODE;
/*菜单操作结构体*/
typedef struct Menu
{
    /*当前正在执行的页面*/
    uint8_t Current_Page ;
    /*当前触发的事件*/
    int KeyEvent ;
} Menu ;
/*菜单初始化*/
void menu_init(struct Menu *handle, uint8_t Page, int EVENT_CODE);
/*获取当前菜单*/
uint8_t Get_Menu(struct Menu *handle);
/*菜单跳转*/
void Set_Menu(struct Menu *handle, uint8_t Page);
/*获取当前发生的事件值*/
int Get_Event_Code(struct Menu *handle);
/*设置当前发生的事件值*/
void Set_Event_Code(struct Menu *handle, int Event_Code);

再来看看.c文件相关函数的实现:

/*菜单初始化*/
void menu_init(struct Menu *handle, uint8_t Page, int EVENT_CODE)
{
    memset(handle, 0, sizeof(struct Menu));
    handle->Current_Page = Page  ;
    handle->KeyEvent = EVENT_CODE ;
}
/*菜单跳转*/
void Set_Menu(struct Menu *handle, uint8_t Page)
{
    handle->Current_Page = Page  ;
}
/*获取当前菜单*/
uint8_t Get_Menu(struct Menu *handle)
{
    return handle->Current_Page ;
}
/*设置当前发生的事件值*/
void Set_Event_Code(struct Menu *handle, int Event_Code)
{
    handle->KeyEvent = Event_Code ;
}
/*获取当前发生的事件值*/
int Get_Event_Code(struct Menu *handle)
{
    return handle->KeyEvent ;
}

非常简单,菜单的操作就在MutilButton的两个文件中添加完成了,接下来看看怎么来使用吧?

1、常规的MultiButton的使用方法

Button button ;
/*读取按键状态*/
uint8_t read_button_pin_status(void)
{
    return HAL_GPIO_ReadPin(KEY_GPIO_Port, KEY_Pin); ;
}

初始化multi_button并注册button event

button_init(&button, read_button_pin_status, 0);
button_attach(&button, SINGLE_CLICK, button_callback);
button_attach(&button, LONG_RRESS_START, button_callback);
button_start(&button);

当前我注册的这个按键的回调函数是同一个函数,而同一个函数做不同的处理:

void button_callback(void *event)
{
    uint8_t button_event = get_button_event(&button) ;
    switch(button_event)
    {
        case SINGLE_CLICK:
            /*当按键发生单击时,注册为ENTER_KEY_SHORT*/
            Set_Event_Code(&menu, ENTER_KEY_SHORT);
            break ;
        case LONG_RRESS_START:
            /*当按键发生长按时,注册为ENTER_KEY_LONG*/
            Set_Event_Code(&menu, ENTER_KEY_LONG);
            break ;
        default:
            break ;
    }
}

2、初始化菜单以及初始化触发事件

Menu   menu ;
//初始化页面为主页面,初始事件为NULL_KEY_EVENT
menu_init(&menu, MAIN_PAGE, NULL_KEY_EVENT);

3、循环调用菜单处理函数

while(1)
{
        /*用户代码*/
        /*......*/
  Menu_Handler(&menu);
  /*......*/
        timer_loop();
   /*用户代码*/
}

注意,这里还需要一个5ms的定时器来调用button_ticks();这样MultiButton才能真正工作起来,我采用的是另外一套开源的软件定时器框架MultiTimer来实现的,这里就不贴出来了,后面直接分享工程实践测试源代码。


菜单处理函数的实现:

/*菜单处理*/
void Menu_Handler(struct Menu *handle)
{
    /*当前是菜单的哪个页面*/
    switch(handle->Current_Page)
    {
        case MAIN_PAGE :
            /*针对注册的键值做相应的处理*/
            main_page_process(handle->KeyEvent);
            break ;
        case LOG_PAGE:
            /*针对注册的键值做相应的处理*/
            log_page_process(handle->KeyEvent);
            break ;
        default:
            break ;
    }
    /*及时将事件清除,防止重复触发*/
    Set_Event_Code(handle, NULL_KEY_EVENT);
}

菜单对应的页面处理函数:

void main_page_process(uint8_t Event_Code)
{
    switch(Event_Code)
    {
      /*当发生事件时,需要的时候做状态切换*/
        case ENTER_KEY_SHORT:
            printf("发生单击,进入页面1\n");
            Set_Menu(&menu, LOG_PAGE);
            break ;
        case ENTER_KEY_LONG:
            printf("在页面0发生长按\n");
            break ;
    }
}
void log_page_process(uint8_t Event_Code)
{
    switch(Event_Code)
    {
        case ENTER_KEY_SHORT:
            printf("在页面1发生单击\n");
            break ;
  /*当发生事件时,需要的时候做状态切换*/
        case ENTER_KEY_LONG:
            printf("发生长按,返回页面0\n");
            Set_Menu(&menu, MAIN_PAGE);
            break ;
    }
}

测试运行结果:

640.png

测试平台:stm32f103c8t6


640.png

测试工程下载:

链接:https://pan.baidu.com/s/124jks9I9uVXmKN3SXHQXvg
提取码:hv3g
复制这段内容后打开百度网盘手机App,操作更方便哦


目录
相关文章
|
9月前
|
iOS开发 MacOS
LabVIEW如何使用热键去触发自定义的事件
LabVIEW如何使用热键去触发自定义的事件
181 1
|
3月前
|
开发者 UED 容器
鸿蒙next版开发:ArkTS组件通用属性(Z序控制)
在HarmonyOS 5.0中,ArkTS引入了Z序控制属性,使开发者能够设置组件的堆叠顺序。本文详细解读了Z序控制的基础知识、zIndex属性及其用途,并提供了示例代码,帮助开发者实现复杂的用户界面和动态交互效果。
82 6
|
3月前
|
UED 开发者
鸿蒙next版开发:ArkTS组件通用属性(禁用控制)
在HarmonyOS 5.0中,ArkTS引入了禁用控制属性,允许开发者控制组件的可用状态,提升用户界面的交互性和响应性。本文详细解读了ArkTS中组件的禁用控制属性,并提供了示例代码,展示了如何使用`disabled`属性来禁用按钮等可交互组件,从而防止用户误操作、引导用户流程和提升用户体验。
113 4
|
3月前
|
开发者
鸿蒙next版开发:ArkTS组件通用属性(菜单控制)
在HarmonyOS 5.0中,ArkTS引入了灵活的菜单控制属性,支持通过长按、点击或鼠标右键触发弹出式菜单,增强用户交互体验。本文详细介绍了bindMenu和bindContextMenu方法,以及MenuItem的配置属性,并提供了示例代码,帮助开发者更好地理解和使用这些功能。
191 1
|
3月前
|
开发者 UED
鸿蒙next版开发:ArkTS组件通用属性(悬浮态效果)
在HarmonyOS 5.0中,ArkTS引入了悬浮态效果的控制属性,使开发者能为组件添加鼠标悬浮时的视觉反馈,增强用户体验。本文详解了hoverEffect属性及其常见效果(Auto、Scale、Highlight、None),并提供了示例代码,展示了如何为按钮设置悬浮效果。通过这些属性,开发者可以实现更生动和互动的界面。
379 1
|
3月前
|
API 开发者 UED
鸿蒙next版开发:ArkTS组件通用属性(焦点控制)
在HarmonyOS 5.0中,ArkTS提供了完善的焦点控制属性,如focusable、defaultFocus、onFocus和onBlur,帮助开发者管理和响应用户界面中的焦点变化。本文详细介绍这些属性,并通过示例代码展示如何使用FocusController类进行焦点管理,提升应用的交互性和无障碍支持。
208 1
|
6月前
|
UED 开发者
哇塞!Uno Platform 数据绑定超全技巧大揭秘!从基础绑定到高级转换,优化性能让你的开发如虎添翼
【8月更文挑战第31天】在开发过程中,数据绑定是连接数据模型与用户界面的关键环节,可实现数据自动更新。Uno Platform 提供了简洁高效的数据绑定方式,使属性变化时 UI 自动同步更新。通过示例展示了基本绑定方法及使用 `Converter` 转换数据的高级技巧,如将年龄转换为格式化字符串。此外,还可利用 `BindingMode.OneTime` 提升性能。掌握这些技巧能显著提高开发效率并优化用户体验。
97 0
|
9月前
|
小程序
Uniapp 解决组件在官方文档不支持的事件上,接收小程序原生组件事件
Uniapp 解决组件在官方文档不支持的事件上,接收小程序原生组件事件
124 0
|
JSON 算法 物联网
第1期 | MultiButton,一个小巧简单易用的事件驱动型按键驱动模块
第1期 | MultiButton,一个小巧简单易用的事件驱动型按键驱动模块
362 0