单片机按键FIFO
本文代碼參考安富萊按鍵FIFO
FIFO數據結構如果不清楚可以參考博文簡單的FIFO
一般的單片機系統,按鍵作為人機交互工具是必不可少的,但是普通的按鍵需要消抖處理,極大的增加了程序開銷,降低系統實時性。
安富萊的FIFO按鍵,無需延時處理消抖,可以記錄按鍵按下、彈起、長按、組合按,并且移植起來也十分方便。之前在做一個項目時,用到一個矩陣鍵盤,移植了這個按鍵FIFO程序,用起來效果很不錯。
主要流程就是開啟一個10ms的定時器中斷,在中斷中掃描按鍵狀態,并對按鍵狀態進行分析消抖處理,如果按鍵動作,將按鍵動作壓入FIFO中,在主循環中讀取FIFO,獲取按鍵狀態。
使用時首先要調用初始化函數,此函數有兩個子函數,分別完成變量初始化和板子硬件初始化。
/* ********************************************************************************************************* * 函 數 名: bsp_InitKey * 功能說明: 初始化按鍵. 該函數被 bsp_Init() 調用。 * 形 參: 無 * 返 回 值: 無 ********************************************************************************************************* */ void bsp_InitKey(void) {bsp_InitKeyVar(); /* 初始化按鍵變量 */bsp_InitKeyHard(); /* 初始化按鍵硬件 */ }移植時需要注意對應IO硬件初始化修改成自己的,這里使用的STC8A單片機,3*4矩陣鍵盤,使用的IO通過宏定義封裝起來。
按鍵參數初始化需要注意設置連發速度來確定對應按鍵是否支持連按。還要自行修改判斷按鍵按下函數和按鍵個數
/*按鍵濾波時間50ms, 單位10ms。只有連續檢測到50ms狀態不變才認為有效,包括彈起和按下兩種事件即使按鍵電路不做硬件濾波,該濾波機制也可以保證可靠地檢測到按鍵事件 */ #define KEY_FILTER_TIME 5 #define KEY_LONG_TIME 100 /* 單位10ms, 持續1秒,認為長按事件 */ #define KEY_COUNT 13 /* 按鍵個數, 12個獨立建 + 1 個組合鍵 */ static KEY_T xdata s_tBtn[KEY_COUNT]; static KEY_FIFO_T xdata s_tKey; /* 按鍵FIFO變量,結構體 *//* 檢測按鍵按下函數 */ static uint8_t IsKeyDown0(void) {P(C1_GPIO_PORT, C1_PIN) = 1; P(C2_GPIO_PORT, C2_PIN) = 1; P(C3_GPIO_PORT, C3_PIN) = 1; P(C4_GPIO_PORT, C4_PIN) = 0; if (P(R2_GPIO_PORT, R2_PIN) == 0) return 1;else return 0;} static uint8_t IsKeyDown1(void) {P(C1_GPIO_PORT, C1_PIN) = 0; P(C2_GPIO_PORT, C2_PIN) = 1; P(C3_GPIO_PORT, C3_PIN) = 1; P(C4_GPIO_PORT, C4_PIN) = 1; NOP(50); if (P(R1_GPIO_PORT, R1_PIN) == 0) return 1;else return 0;} static uint8_t IsKeyDown2(void) {P(C1_GPIO_PORT, C1_PIN) = 0; P(C2_GPIO_PORT, C2_PIN) = 1; P(C3_GPIO_PORT, C3_PIN) = 1; P(C4_GPIO_PORT, C4_PIN) = 1; if (P(R2_GPIO_PORT, R2_PIN) == 0) return 1;else return 0;} static uint8_t IsKeyDown3(void) {P(C1_GPIO_PORT, C1_PIN) = 0; P(C2_GPIO_PORT, C2_PIN) = 1; P(C3_GPIO_PORT, C3_PIN) = 1; P(C4_GPIO_PORT, C4_PIN) = 1; if (P(R3_GPIO_PORT, R3_PIN) == 0) return 1;else return 0;} static uint8_t IsKeyDown4(void) {P(C1_GPIO_PORT, C1_PIN) = 1; P(C2_GPIO_PORT, C2_PIN) = 0; P(C3_GPIO_PORT, C3_PIN) = 1; P(C4_GPIO_PORT, C4_PIN) = 1; if (P(R1_GPIO_PORT, R1_PIN) == 0) return 1;else return 0;} static uint8_t IsKeyDown5(void) {P(C1_GPIO_PORT, C1_PIN) = 1; P(C2_GPIO_PORT, C2_PIN) = 0; P(C3_GPIO_PORT, C3_PIN) = 1; P(C4_GPIO_PORT, C4_PIN) = 1; if (P(R2_GPIO_PORT, R2_PIN) == 0) return 1;else return 0;} static uint8_t IsKeyDown6(void) {P(C1_GPIO_PORT, C1_PIN) = 1; P(C2_GPIO_PORT, C2_PIN) = 0; P(C3_GPIO_PORT, C3_PIN) = 1; P(C4_GPIO_PORT, C4_PIN) = 1; if (P(R3_GPIO_PORT, R3_PIN) == 0) return 1;else return 0;} static uint8_t IsKeyDown7(void) {P(C1_GPIO_PORT, C1_PIN) = 1; P(C2_GPIO_PORT, C2_PIN) = 1; P(C3_GPIO_PORT, C3_PIN) = 0; P(C4_GPIO_PORT, C4_PIN) = 1; if (P(R1_GPIO_PORT, R1_PIN) == 0) return 1;else return 0;} static uint8_t IsKeyDown8(void) {P(C1_GPIO_PORT, C1_PIN) = 1; P(C2_GPIO_PORT, C2_PIN) = 1; P(C3_GPIO_PORT, C3_PIN) = 0; P(C4_GPIO_PORT, C4_PIN) = 1; if (P(R2_GPIO_PORT, R2_PIN) == 0) return 1;else return 0;} static uint8_t IsKeyDown9(void) {P(C1_GPIO_PORT, C1_PIN) = 1; P(C2_GPIO_PORT, C2_PIN) = 1; P(C3_GPIO_PORT, C3_PIN) = 0; P(C4_GPIO_PORT, C4_PIN) = 1; if (P(R3_GPIO_PORT, R3_PIN) == 0) return 1;else return 0;} static uint8_t IsKeyDown10(void) {P(C1_GPIO_PORT, C1_PIN) = 1; P(C2_GPIO_PORT, C2_PIN) = 1; P(C3_GPIO_PORT, C3_PIN) = 1; P(C4_GPIO_PORT, C4_PIN) = 0; NOP(50); if (P(R1_GPIO_PORT, R1_PIN) == 0) return 1;else return 0;} static uint8_t IsKeyDown11(void) {P(C1_GPIO_PORT, C1_PIN) = 1; P(C2_GPIO_PORT, C2_PIN) = 1; P(C3_GPIO_PORT, C3_PIN) = 1; P(C4_GPIO_PORT, C4_PIN) = 0; if (P(R3_GPIO_PORT, R3_PIN) == 0) return 1;else return 0;}/* 組合按鍵 key1 && key2 */ static uint8_t IsKeyDown12(void) { return IsKeyDown1() && IsKeyDown2();} /* ********************************************************************************************************* * 函 數 名: bsp_InitKeyVar * 功能說明: 初始化按鍵變量 * 形 參: 無 * 返 回 值: 無 ********************************************************************************************************* */ static void bsp_InitKeyVar(void) {uint8_t xdata i;/* 對按鍵FIFO讀寫指針清零 */fifo_init(&s_tKey);/* 給每個按鍵結構體成員變量賦一組缺省值 */for (i = 0; i < KEY_COUNT; i++){s_tBtn[i].LongTime = KEY_LONG_TIME; /* 長按時間 0 表示不檢測長按鍵事件 */s_tBtn[i].Count = KEY_FILTER_TIME / 2; /* 計數器設置為濾波時間的一半 */s_tBtn[i].State = 0; /* 按鍵缺省狀態,0為未按下 *///s_tBtn[i].KeyCodeDown = 3 * i + 1; /* 按鍵按下的鍵值代碼 *///s_tBtn[i].KeyCodeUp = 3 * i + 2; /* 按鍵彈起的鍵值代碼 *///s_tBtn[i].KeyCodeLong = 3 * i + 3; /* 按鍵被持續按下的鍵值代碼 */s_tBtn[i].RepeatSpeed = 0; /* 按鍵連發的速度,0表示不支持連發 */s_tBtn[i].RepeatCount = 0; /* 連發計數器 */}/* 判斷按鍵按下的函數 */s_tBtn[0].IsKeyDownFunc = IsKeyDown0;s_tBtn[1].IsKeyDownFunc = IsKeyDown1;s_tBtn[2].IsKeyDownFunc = IsKeyDown2;s_tBtn[3].IsKeyDownFunc = IsKeyDown3;s_tBtn[4].IsKeyDownFunc = IsKeyDown4;s_tBtn[5].IsKeyDownFunc = IsKeyDown5;s_tBtn[6].IsKeyDownFunc = IsKeyDown6;s_tBtn[7].IsKeyDownFunc = IsKeyDown7;s_tBtn[8].IsKeyDownFunc = IsKeyDown8;s_tBtn[9].IsKeyDownFunc = IsKeyDown9;s_tBtn[10].IsKeyDownFunc = IsKeyDown10;s_tBtn[11].IsKeyDownFunc = IsKeyDown11;/* 組合按鍵 */s_tBtn[12].IsKeyDownFunc = IsKeyDown12; }按鍵掃描函數,需要開啟一個10ms的定時器中斷,在中斷中對按鍵狀態進行掃描。
/* ********************************************************************************************************* * 函 數 名: bsp_DetectKey * 功能說明: 檢測一個按鍵。非阻塞狀態,必須被周期性的調用。 * 形 參: 按鍵結構變量指針 * 返 回 值: 無 ********************************************************************************************************* */ static void bsp_DetectKey(uint8_t i) {KEY_T xdata *pBtn;/*如果沒有初始化按鍵函數,則報錯if (s_tBtn[i].IsKeyDownFunc == 0){printf("Fault : DetectButton(), s_tBtn[i].IsKeyDownFunc undefine");}*/pBtn = &s_tBtn[i];if (pBtn->IsKeyDownFunc()){if (pBtn->Count < KEY_FILTER_TIME){pBtn->Count = KEY_FILTER_TIME;}else if(pBtn->Count < 2 * KEY_FILTER_TIME){pBtn->Count++;}else{if (pBtn->State == 0){pBtn->State = 1;/* 發送按鈕按下的消息 */bsp_PutKey((uint8_t)(3 * i + 1));}if (pBtn->LongTime > 0){if (pBtn->LongCount < pBtn->LongTime){/* 發送按鈕持續按下的消息 */if (++pBtn->LongCount == pBtn->LongTime){/* 鍵值放入按鍵FIFO */bsp_PutKey((uint8_t)(3 * i + 3));}}else{if (pBtn->RepeatSpeed > 0){if (++pBtn->RepeatCount >= pBtn->RepeatSpeed){pBtn->RepeatCount = 0;/* 常按鍵后,每隔10ms發送1個按鍵 */bsp_PutKey((uint8_t)(3 * i + 1));}}}}}}else{if(pBtn->Count > KEY_FILTER_TIME){pBtn->Count = KEY_FILTER_TIME;}else if(pBtn->Count != 0){pBtn->Count--;}else{if (pBtn->State == 1){pBtn->State = 0;/* 發送按鈕彈起的消息 */bsp_PutKey((uint8_t)(3 * i + 2));}}pBtn->LongCount = 0;pBtn->RepeatCount = 0;} } /* ********************************************************************************************************* * 函 數 名: bsp_KeyScan * 功能說明: 掃描所有按鍵。非阻塞,被systick中斷周期性的調用 * 形 參: 無 * 返 回 值: 無 ********************************************************************************************************* */ void bsp_KeyScan(void) {uint8_t xdata i;for (i = 0; i < KEY_COUNT; i++){bsp_DetectKey(i);} }在主函數中,非堵塞方式獲取按鍵鍵值
/* ********************************************************************************************************* * 函 數 名: bsp_GetKey * 功能說明: 從按鍵FIFO緩沖區讀取一個鍵值。 * 形 參: 無 * 返 回 值: 按鍵代碼 ********************************************************************************************************* */ uint8_t bsp_GetKey(void) {uint8_t xdata ret;if (fifo_pop(&s_tKey, &ret) == 1){return KEY_NONE;}else{return ret;} }完整代碼如下
bsp_key.h
bsp_key.c
/*!* @file BSP_KEY.c** @brief 按鍵驅動文件** @company ** @author 不咸不要錢** @note 無** @version ** @date 2019/10/18 星期五*/ #include "bsp_key.h" #include "lq_gpio.h" #include "stdio.h"static KEY_T xdata s_tBtn[KEY_COUNT]; static KEY_FIFO_T xdata s_tKey; /* 按鍵FIFO變量,結構體 */static void bsp_InitKeyVar(void); static void bsp_InitKeyHard(void); static void bsp_DetectKey(uint8_t i);static uint8_t IsKeyDown0(void) {P(C1_GPIO_PORT, C1_PIN) = 1; P(C2_GPIO_PORT, C2_PIN) = 1; P(C3_GPIO_PORT, C3_PIN) = 1; P(C4_GPIO_PORT, C4_PIN) = 0; if (P(R2_GPIO_PORT, R2_PIN) == 0) return 1;else return 0;} static uint8_t IsKeyDown1(void) {P(C1_GPIO_PORT, C1_PIN) = 0; P(C2_GPIO_PORT, C2_PIN) = 1; P(C3_GPIO_PORT, C3_PIN) = 1; P(C4_GPIO_PORT, C4_PIN) = 1; NOP(50); if (P(R1_GPIO_PORT, R1_PIN) == 0) return 1;else return 0;} static uint8_t IsKeyDown2(void) {P(C1_GPIO_PORT, C1_PIN) = 0; P(C2_GPIO_PORT, C2_PIN) = 1; P(C3_GPIO_PORT, C3_PIN) = 1; P(C4_GPIO_PORT, C4_PIN) = 1; if (P(R2_GPIO_PORT, R2_PIN) == 0) return 1;else return 0;} static uint8_t IsKeyDown3(void) {P(C1_GPIO_PORT, C1_PIN) = 0; P(C2_GPIO_PORT, C2_PIN) = 1; P(C3_GPIO_PORT, C3_PIN) = 1; P(C4_GPIO_PORT, C4_PIN) = 1; if (P(R3_GPIO_PORT, R3_PIN) == 0) return 1;else return 0;} static uint8_t IsKeyDown4(void) {P(C1_GPIO_PORT, C1_PIN) = 1; P(C2_GPIO_PORT, C2_PIN) = 0; P(C3_GPIO_PORT, C3_PIN) = 1; P(C4_GPIO_PORT, C4_PIN) = 1; if (P(R1_GPIO_PORT, R1_PIN) == 0) return 1;else return 0;} static uint8_t IsKeyDown5(void) {P(C1_GPIO_PORT, C1_PIN) = 1; P(C2_GPIO_PORT, C2_PIN) = 0; P(C3_GPIO_PORT, C3_PIN) = 1; P(C4_GPIO_PORT, C4_PIN) = 1; if (P(R2_GPIO_PORT, R2_PIN) == 0) return 1;else return 0;} static uint8_t IsKeyDown6(void) {P(C1_GPIO_PORT, C1_PIN) = 1; P(C2_GPIO_PORT, C2_PIN) = 0; P(C3_GPIO_PORT, C3_PIN) = 1; P(C4_GPIO_PORT, C4_PIN) = 1; if (P(R3_GPIO_PORT, R3_PIN) == 0) return 1;else return 0;} static uint8_t IsKeyDown7(void) {P(C1_GPIO_PORT, C1_PIN) = 1; P(C2_GPIO_PORT, C2_PIN) = 1; P(C3_GPIO_PORT, C3_PIN) = 0; P(C4_GPIO_PORT, C4_PIN) = 1; if (P(R1_GPIO_PORT, R1_PIN) == 0) return 1;else return 0;} static uint8_t IsKeyDown8(void) {P(C1_GPIO_PORT, C1_PIN) = 1; P(C2_GPIO_PORT, C2_PIN) = 1; P(C3_GPIO_PORT, C3_PIN) = 0; P(C4_GPIO_PORT, C4_PIN) = 1; if (P(R2_GPIO_PORT, R2_PIN) == 0) return 1;else return 0;} static uint8_t IsKeyDown9(void) {P(C1_GPIO_PORT, C1_PIN) = 1; P(C2_GPIO_PORT, C2_PIN) = 1; P(C3_GPIO_PORT, C3_PIN) = 0; P(C4_GPIO_PORT, C4_PIN) = 1; if (P(R3_GPIO_PORT, R3_PIN) == 0) return 1;else return 0;} static uint8_t IsKeyDown10(void) {P(C1_GPIO_PORT, C1_PIN) = 1; P(C2_GPIO_PORT, C2_PIN) = 1; P(C3_GPIO_PORT, C3_PIN) = 1; P(C4_GPIO_PORT, C4_PIN) = 0; NOP(50); if (P(R1_GPIO_PORT, R1_PIN) == 0) return 1;else return 0;} static uint8_t IsKeyDown11(void) {P(C1_GPIO_PORT, C1_PIN) = 1; P(C2_GPIO_PORT, C2_PIN) = 1; P(C3_GPIO_PORT, C3_PIN) = 1; P(C4_GPIO_PORT, C4_PIN) = 0; if (P(R3_GPIO_PORT, R3_PIN) == 0) return 1;else return 0;}/* 組合按鍵 key1 && key2 */ static uint8_t IsKeyDown12(void) { return IsKeyDown1() && IsKeyDown2();}/*!* @brief fifo初始化** @param fifo_t : FIFO** @return 無** @note 無** @see fifo_t tempFifo;* fifo_init(tempFifo); //fifo初始化** @date 2020/5/21*/ void fifo_init(fifo_t *fifo) {fifo->fifoLen = 0;fifo->fifoRead = 0;fifo->fifoWrite = 0; }/*!* @brief fifo壓入數據** @param fifo_t : FIFO* @param data : 入棧數據** @return 0 :成功 1 :失敗** @note FIFO_COVER 宏定義進行判斷緩沖區滿后的操作** @see fifo_t tempFifo;* fifo_push(tempFifo, 120); //fifo中壓入一個數據** @date 2020/5/21*/ uint8_t fifo_push(fifo_t *fifo, FIFO_TYPE dat) {fifo->fifoLen++;/* 判斷緩沖區是否已滿 */if(fifo->fifoLen > FIFO_SIZE){fifo->fifoLen = FIFO_SIZE;#if FIFO_COVERreturn 1; #elseif(++fifo->fifoRead >= FIFO_SIZE){fifo->fifoRead = 0;} #endif}fifo->buff[fifo->fifoWrite] = dat;if(++fifo->fifoWrite >= FIFO_SIZE){fifo->fifoWrite = 0;}return 0;}/*!* @brief fifo彈出數據** @param fifo_t : FIFO* @param data : 出棧數據** @return 0 :成功 1 :失敗** @note 無** @see fifo_t tempFifo;* FIFO_TYPE tempData;* fifo_push(tempFifo, 120); //fifo中壓入一個數據* fifo_pop(tempFifo, &tempData); //fifo中彈出一個數據** @date 2020/5/21*/ uint8_t fifo_pop(fifo_t *fifo, FIFO_TYPE *dat) {/* 緩沖區為空 */if(fifo->fifoLen == 0){return 1;}fifo->fifoLen--;*dat = fifo->buff[fifo->fifoRead];if(++fifo->fifoRead >= FIFO_SIZE){fifo->fifoRead = 0;}return 0; }/* ********************************************************************************************************* * 函 數 名: bsp_InitKey * 功能說明: 初始化按鍵. 該函數被 bsp_Init() 調用。 * 形 參: 無 * 返 回 值: 無 ********************************************************************************************************* */ void bsp_InitKey(void) {bsp_InitKeyVar(); /* 初始化按鍵變量 */bsp_InitKeyHard(); /* 初始化按鍵硬件 */ }/* ********************************************************************************************************* * 函 數 名: bsp_PutKey * 功能說明: 將1個鍵值壓入按鍵FIFO緩沖區。可用于模擬一個按鍵。 * 形 參: _KeyCode : 按鍵代碼 * 返 回 值: 無 ********************************************************************************************************* */ void bsp_PutKey(uint8_t _KeyCode) {fifo_push(&s_tKey, _KeyCode); }/* ********************************************************************************************************* * 函 數 名: bsp_GetKey * 功能說明: 從按鍵FIFO緩沖區讀取一個鍵值。 * 形 參: 無 * 返 回 值: 按鍵代碼 ********************************************************************************************************* */ uint8_t bsp_GetKey(void) {uint8_t xdata ret;if (fifo_pop(&s_tKey, &ret) == 1){return KEY_NONE;}else{return ret;} }/* ********************************************************************************************************* * 函 數 名: bsp_GetKeyState * 功能說明: 讀取按鍵的狀態 * 形 參: _ucKeyID : 按鍵ID,從0開始 * 返 回 值: 1 表示按下, 0 表示未按下 ********************************************************************************************************* */ uint8_t bsp_GetKeyState(KEY_ID_E _ucKeyID) {return s_tBtn[_ucKeyID].State; }/* ********************************************************************************************************* * 函 數 名: bsp_SetKeyParam * 功能說明: 設置按鍵參數 * 形 參:_ucKeyID : 按鍵ID,從0開始 * _LongTime : 長按事件時間 * _RepeatSpeed : 連發速度 * 返 回 值: 無 ********************************************************************************************************* */ void bsp_SetKeyParam(uint8_t _ucKeyID, uint16_t _LongTime, uint8_t _RepeatSpeed) {s_tBtn[_ucKeyID].LongTime = _LongTime; /* 長按時間 0 表示不檢測長按鍵事件 */s_tBtn[_ucKeyID].RepeatSpeed = _RepeatSpeed; /* 按鍵連發的速度,0表示不支持連發 */s_tBtn[_ucKeyID].RepeatCount = 0; /* 連發計數器 */ }/* ********************************************************************************************************* * 函 數 名: bsp_ClearKey * 功能說明: 清空按鍵FIFO緩沖區 * 形 參:無 * 返 回 值: 按鍵代碼 ********************************************************************************************************* */ void bsp_ClearKey(void) {s_tKey.fifoRead = s_tKey.fifoWrite; }/* ********************************************************************************************************* * 函 數 名: bsp_InitKeyHard * 功能說明: 配置按鍵對應的GPIO * 形 參: 無 * 返 回 值: 無 ********************************************************************************************************* */ static void bsp_InitKeyHard(void) {//KEY 初始化PIN_InitPushPull(C1_GPIO_PORT, C1_PIN);PIN_InitPushPull(C2_GPIO_PORT, C2_PIN);PIN_InitPushPull(C3_GPIO_PORT, C3_PIN);PIN_InitPushPull(C4_GPIO_PORT, C4_PIN);PIN_InitOpenDrain(R1_GPIO_PORT, R1_PIN);PIN_InitOpenDrain(R2_GPIO_PORT, R2_PIN);PIN_InitOpenDrain(R3_GPIO_PORT, R3_PIN);}/* ********************************************************************************************************* * 函 數 名: bsp_InitKeyVar * 功能說明: 初始化按鍵變量 * 形 參: 無 * 返 回 值: 無 ********************************************************************************************************* */ static void bsp_InitKeyVar(void) {uint8_t xdata i;/* 對按鍵FIFO讀寫指針清零 */fifo_init(&s_tKey);/* 給每個按鍵結構體成員變量賦一組缺省值 */for (i = 0; i < KEY_COUNT; i++){s_tBtn[i].LongTime = KEY_LONG_TIME; /* 長按時間 0 表示不檢測長按鍵事件 */s_tBtn[i].Count = KEY_FILTER_TIME / 2; /* 計數器設置為濾波時間的一半 */s_tBtn[i].State = 0; /* 按鍵缺省狀態,0為未按下 *///s_tBtn[i].KeyCodeDown = 3 * i + 1; /* 按鍵按下的鍵值代碼 *///s_tBtn[i].KeyCodeUp = 3 * i + 2; /* 按鍵彈起的鍵值代碼 *///s_tBtn[i].KeyCodeLong = 3 * i + 3; /* 按鍵被持續按下的鍵值代碼 */s_tBtn[i].RepeatSpeed = 0; /* 按鍵連發的速度,0表示不支持連發 */s_tBtn[i].RepeatCount = 0; /* 連發計數器 */}/* 判斷按鍵按下的函數 */s_tBtn[0].IsKeyDownFunc = IsKeyDown0;s_tBtn[1].IsKeyDownFunc = IsKeyDown1;s_tBtn[2].IsKeyDownFunc = IsKeyDown2;s_tBtn[3].IsKeyDownFunc = IsKeyDown3;s_tBtn[4].IsKeyDownFunc = IsKeyDown4;s_tBtn[5].IsKeyDownFunc = IsKeyDown5;s_tBtn[6].IsKeyDownFunc = IsKeyDown6;s_tBtn[7].IsKeyDownFunc = IsKeyDown7;s_tBtn[8].IsKeyDownFunc = IsKeyDown8;s_tBtn[9].IsKeyDownFunc = IsKeyDown9;s_tBtn[10].IsKeyDownFunc = IsKeyDown10;s_tBtn[11].IsKeyDownFunc = IsKeyDown11;/* 組合按鍵 */s_tBtn[12].IsKeyDownFunc = IsKeyDown12; }/* ********************************************************************************************************* * 函 數 名: bsp_DetectKey * 功能說明: 檢測一個按鍵。非阻塞狀態,必須被周期性的調用。 * 形 參: 按鍵結構變量指針 * 返 回 值: 無 ********************************************************************************************************* */ static void bsp_DetectKey(uint8_t i) {KEY_T xdata *pBtn;/*如果沒有初始化按鍵函數,則報錯if (s_tBtn[i].IsKeyDownFunc == 0){printf("Fault : DetectButton(), s_tBtn[i].IsKeyDownFunc undefine");}*/pBtn = &s_tBtn[i];if (pBtn->IsKeyDownFunc()){if (pBtn->Count < KEY_FILTER_TIME){pBtn->Count = KEY_FILTER_TIME;}else if(pBtn->Count < 2 * KEY_FILTER_TIME){pBtn->Count++;}else{if (pBtn->State == 0){pBtn->State = 1;/* 發送按鈕按下的消息 */bsp_PutKey((uint8_t)(3 * i + 1));P27 = 1; //開啟蜂鳴器}if (pBtn->LongTime > 0){if (pBtn->LongCount < pBtn->LongTime){/* 發送按鈕持續按下的消息 */if (++pBtn->LongCount == pBtn->LongTime){/* 鍵值放入按鍵FIFO */bsp_PutKey((uint8_t)(3 * i + 3));}}else{if (pBtn->RepeatSpeed > 0){if (++pBtn->RepeatCount >= pBtn->RepeatSpeed){pBtn->RepeatCount = 0;/* 常按鍵后,每隔10ms發送1個按鍵 */bsp_PutKey((uint8_t)(3 * i + 1));}}}}}}else{if(pBtn->Count > KEY_FILTER_TIME){pBtn->Count = KEY_FILTER_TIME;}else if(pBtn->Count != 0){pBtn->Count--;}else{if (pBtn->State == 1){pBtn->State = 0;/* 發送按鈕彈起的消息 */bsp_PutKey((uint8_t)(3 * i + 2));P27 = 0; //關閉蜂鳴器}}pBtn->LongCount = 0;pBtn->RepeatCount = 0;} }/* ********************************************************************************************************* * 函 數 名: bsp_KeyScan * 功能說明: 掃描所有按鍵。非阻塞,被systick中斷周期性的調用 * 形 參: 無 * 返 回 值: 無 ********************************************************************************************************* */ void bsp_KeyScan(void) {uint8_t xdata i;for (i = 0; i < KEY_COUNT; i++){bsp_DetectKey(i);} }思考
這個按鍵底層驅動可以檢測到底層按下彈起,是否支持雙擊?
雙擊實現可以基于此底層驅動,不需要修改代碼,只需要在修改應用層,比如檢測到某按鍵按下,開啟一個軟件定時器,在定時器定時結束前,再次檢測到按下,則為雙擊。
總結
- 上一篇: 智能车声标定位相关算法优化
- 下一篇: systick定时器 延时计时