| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261 |
- /*
- * @File: multi_button.c
- * @Author: Zibin Zheng
- * @Date: 2018-01-23 20:36:01
- *
- * @LICENSE: Copyright (c) 2016 Zibin Zheng <znbin@qq.com>
- * All rights reserved.
- *
- * @NOTE: The original author of MultiButton is Zibin Zheng.
- * Please contact the original author for authorization
- * before use.I(liu2guang) only adapt the library to
- * rt-thread and fix some bugs. I(liu2guang) am not
- * responsible for the authorization of the library.
- *
- * Change Logs:
- * Date Author Notes
- * 2018-01-23 liu2guang Adapter rtthread and Fix the bug.
- */
- #include "multi_button.h"
- #define EVENT_CB(ev) if(handle->cb[ev]) handle->cb[ev]((button*)handle)
- static struct button* head_handle = NULL;
- /**
- * @brief Initializes the button struct handle.
- * @param handle: the button handle strcut.
- * @param pin_level: read the pin of the connet button level.
- * @param active_level: pin pressed level.
- * @retval None
- */
- void button_init(struct button* handle, uint8_t(*pin_level)(void), uint8_t active_level)
- {
- memset(handle, 0, sizeof(struct button));
- handle->event = (uint8_t)NONE_PRESS;
- handle->hal_button_Level = pin_level;
- handle->button_level = handle->hal_button_Level();
- handle->active_level = active_level;
- }
- /**
- * @brief Attach the button event callback function.
- * @param handle: the button handle strcut.
- * @param event: trigger event type.
- * @param cb: callback function.
- * @retval None
- */
- void button_attach(struct button* handle, PressEvent event, BtnCallback cb)
- {
- handle->cb[event] = cb;
- }
- /**
- * @brief Inquire the button event happen.
- * @param handle: the button handle strcut.
- * @retval button event.
- */
- PressEvent get_button_event(struct button* handle)
- {
- return (PressEvent)(handle->event);
- }
- /**
- * @brief button driver core function, driver state machine.
- * @param handle: the button handle strcut.
- * @retval None
- */
- void button_handler(struct button* handle)
- {
- uint8_t read_gpio_level = handle->hal_button_Level();
- //ticks counter working..
- if((handle->state) > 0)
- {
- handle->ticks++;
- }
- /*------------button debounce handle---------------*/
- if(read_gpio_level != handle->button_level)
- {
- //not equal to prev one
- //continue read 3 times same new level change
- if(++(handle->debounce_cnt) >= DEBOUNCE_TICKS)
- {
- handle->button_level = read_gpio_level;
- handle->debounce_cnt = 0;
- }
- }
- else
- {
- // leved not change ,counter reset.
- handle->debounce_cnt = 0;
- }
- /*-----------------State machine-------------------*/
- switch (handle->state)
- {
- case 0:
- if(handle->button_level == handle->active_level)
- {
- handle->event = (uint8_t)PRESS_DOWN;
- EVENT_CB(PRESS_DOWN);
- handle->ticks = 0;
- handle->repeat = 1;
- handle->state = 1;
- }
- else
- {
- handle->event = (uint8_t)NONE_PRESS;
- }
- break;
- case 1:
- if(handle->button_level != handle->active_level)
- {
- handle->event = (uint8_t)PRESS_UP;
- EVENT_CB(PRESS_UP);
- handle->ticks = 0;
- handle->state = 2;
- }
- else if(handle->ticks > LONG_TICKS)
- {
- handle->event = (uint8_t)LONG_PRESS_START;
- EVENT_CB(LONG_PRESS_START);
- handle->state = 5;
- }
- break;
- case 2:
- if(handle->button_level == handle->active_level)
- {
- handle->event = (uint8_t)PRESS_DOWN;
- EVENT_CB(PRESS_DOWN);
- handle->repeat++;
- EVENT_CB(PRESS_REPEAT);
- handle->ticks = 0;
- handle->state = 3;
- }
- else if(handle->ticks > SHORT_TICKS)
- {
- if(handle->repeat == 1)
- {
- handle->event = (uint8_t)SINGLE_CLICK;
- EVENT_CB(SINGLE_CLICK);
- }
- else if(handle->repeat == 2)
- {
- handle->event = (uint8_t)DOUBLE_CLICK;
- EVENT_CB(DOUBLE_CLICK);
- }
- handle->state = 0;
- }
- break;
- case 3:
- if(handle->button_level != handle->active_level)
- {
- handle->event = (uint8_t)PRESS_UP;
- EVENT_CB(PRESS_UP);
- if(handle->ticks < SHORT_TICKS)
- {
- handle->ticks = 0;
- handle->state = 2;
- }
- else
- {
- handle->state = 0;
- }
- }
- else if(handle->ticks > SHORT_TICKS)
- {
- handle->state = 0;
- }
- break;
- case 5:
- if(handle->button_level == handle->active_level)
- {
- handle->event = (uint8_t)LONG_PRESS_HOLD;
- if (handle->ticks % LONG_HOLD_CYC == 0)
- {
- EVENT_CB(LONG_PRESS_HOLD);
- }
- }
- else
- {
- handle->event = (uint8_t)PRESS_UP;
- EVENT_CB(PRESS_UP);
- handle->state = 0;
- }
- break;
- }
- }
- /**
- * @brief Start the button work, add the handle into work list.
- * @param handle: target handle strcut.
- * @retval 0: succeed. -1: already exist.
- */
- int button_start(struct button* handle)
- {
- struct button* target = head_handle;
- while(target)
- {
- if(target == handle)
- {
- return -1; //already exist.
- }
- target = target->next;
- }
- handle->next = head_handle;
- head_handle = handle;
- return 0;
- }
- /**
- * @brief Stop the button work, remove the handle off work list.
- * @param handle: target handle strcut.
- * @retval None
- */
- void button_stop(struct button* handle)
- {
- struct button** curr;
- for(curr = &head_handle; *curr;)
- {
- struct button* entry = *curr;
- if (entry == handle)
- {
- *curr = entry->next;
- return;
- }
- else
- {
- curr = &entry->next;
- }
- }
- }
- /**
- * @brief background ticks, timer repeat invoking interval 5ms.
- * @param None.
- * @retval None
- */
- void button_ticks(void)
- {
- struct button* target;
- for(target = head_handle; target != NULL; target = target->next)
- {
- button_handler(target);
- }
- }
|