前言
2 i: d! S# ]$ ^$ i编程开发环境:STM32CubeIDE5 N( c9 g. F) K. V
按键的单击、双击、长按等在MCU编程中是比较常见且常用的事件,本文章基于STM32来实现,具体思路用在其他MCU也是如此。3 ~8 x3 o5 f# \8 F
2 C1 @# C( {9 b3 ^" u+ G. g+ O8 G8 {具体思路3 o! j$ W' ]0 r( K
初始化一个全局标记
' V& N6 A( E" N$ \" H ]0 s按键中断事件发生后置位标记
; V- n( ^/ P; n3 K1 y1 ^2 Swhile死循环中一直检测这个标记,如果被置位那么进行消抖,然后再次检测连接KEY的IO是否处于按下状态,如是则认为本次按键有效1 z6 r, s7 I) `8 g2 v3 p
第一次按键事件有效后,启动定时器定时300ms,在此定时期间内如果有二次按下那么就是双击,如果没有按下,等到300ms定时时间到后读取IO电平,如果处于松开状态那么本次就是单击事件,如果还是按下状态那么就再次启动700定时器,700ms过后再次读取IO电平是否处于按下状态,如是那么就是长按。9 Z" T; @! [: z: S" b4 F
工程配置
' j& R6 U4 }5 B" [2 u* y6 n我是使用STM32CUBEMX配置生成的工程,配置按键:9 v' t* }( [/ j' N1 M
; X& x. z5 @: `9 [' v" t ?
, c$ J! I z, }5 g
7 {8 X/ G I; O3 j7 r( rSTM32的定时器有向上、向下计数模式,向上计数就是从0计数到N然后触发中断,向下计数模式就是倒计时到0,由于我们需要二次启动定时器来检测长按,所以再次配置为向下计数模式,分频为7199,计数3000就是定时300ms:
* p. K: R k. _# ]$ K" Z# y* [+ v% g9 t
9 C) P8 c4 O* f/ {3 t6 }* x% `0 m
' P/ V7 T8 e ?: @4 ^
中断配置:
5 e; o' U3 {9 n) M% P' H) `1 \. U6 ?5 C e8 q5 U
3 D# e* l2 Y) }/ n# r7 i6 {7 m
: O* O, o/ w* _* a5 h, `代码实现- ~2 Q+ n4 \& E, N8 r
/ Z, ^% E* S7 Q5 E o7 R- /*
6 o3 [' `. `2 l( m - * key_state位说明:
8 x9 P; m8 `2 |) c B$ Y" q - *
D. N6 j* d7 K) b - * bit[15]:按键事件完成标志位
5 a, ]' A2 S& d* J' x, v$ b - * bit[14]:按键中断触发标志位
: g. I# p D9 `8 J+ d - * bit[13]:长按事件触发需要启动第二次定时器标志位
) I* g" n, W& [4 G5 O - * bit[12~0]: 按键计数,=1:单击 =2:双击 =3:长按) G$ j5 U& q% e; [/ j1 i
- */& N5 S$ d- _( t6 Z4 [
- __IO uint16_t key_state = 0;1 j0 S+ Q+ J( h0 W8 Q* r
- $ X' h; I- W5 ^1 Z2 R8 ^
- int __io_putchar(int ch)
# {! \( p: u5 A$ ?7 Y9 R - {' Q1 c& Z& o* T
- HAL_UART_Transmit(&huart1, (uint8_t*) &ch, 1, 100);
( S* s" B/ ~3 H( A6 A* @ - return ch;% {. r& p+ ?, l! ?! Z u/ H) W1 ~: ]
- }7 T1 ~( z/ |- c( A0 t. Z9 y
- ; Y7 e' {6 F" Z5 q% I
- void TIM4_Stop(void)$ n* g1 f( M# J1 T B
- {+ z) H2 ]2 Z% @; \1 L) ^
- HAL_TIM_Base_Stop_IT(&htim4);5 s4 O( i1 x* B4 y2 Q
- }
/ a0 O% s' N+ L' L - F! X. e# q) O) p |: {3 K
- void TIM4_Start(uint16_t timeout)! \* |8 ]$ I7 L1 d& ?
- {
' W! q4 H/ s5 s1 t6 N - __HAL_TIM_SET_COUNTER(&htim4, timeout);
W2 ?) Q% V- s3 _ - HAL_TIM_Base_Start_IT(&htim4);
5 Y, `8 N* f& x6 k5 N4 _0 S4 U; Q - }
9 N1 C2 g$ V5 v4 R5 ^
* B7 {8 d2 X7 Q0 Q S+ h* E6 i- void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)1 t& g- \ c; D6 j1 ]9 ?0 M7 B% o ?" h
- {
3 a4 f( |* y2 J& E( }4 R4 y - if(htim->Instance == htim4.Instance)/ V/ J$ C ?( [# B) F/ T' O
- {
6 n! Y$ y1 C8 J/ w4 U; f# d" @ - TIM4_Stop();9 J& S! h2 q9 F- ]2 U* ~# W, C
- if(key_state & 0X2000) // 如果是700ms的超时
7 Y1 H; H4 G; R& Y9 v4 o( h - {
0 ~ D. ]) T8 [5 _) e* W - // 再次检测按键如果是按下状态,那么就是长按事件,否则事件无效
b0 z0 F' n% n8 N% A# h - if(HAL_GPIO_ReadPin(KEY_GPIO_Port, KEY_Pin) == GPIO_PIN_RESET)
3 g; x% l7 E6 a, ?. B: H - {+ {6 X }6 g& N8 _
- key_state = 0X8003;& n6 P ^: @ I* j' u4 y2 ]4 D
- }else
; z4 n% D. M& H( g2 |- _6 a7 T - {
) I+ {8 k, Q, S& K# x7 t& m - key_state = 0X8000;
; `6 d# C$ @, { {# c - }
9 c; {$ Y# e: ^3 p' C: L - }
) _$ p+ }* \4 T. Q: K( M7 i) u - else // 如果是300ms的超时
2 q! x/ b* `- J8 K - {9 M5 N( p/ w- f5 E6 H( ^3 m+ d r1 [
- // 300ms时间到再次检测按键状态如果仍处于按下状态,那么就启动700ms的超时,0 B U! z3 H4 E
- // 700ms时间到后再次检测按键如果是按下状态,那么就是长按事件了。
) H# ]% U& D2 P6 C! N6 L - if(HAL_GPIO_ReadPin(KEY_GPIO_Port, KEY_Pin) == GPIO_PIN_RESET) // 按键处于按下状态. N: f T) u$ S+ P5 ~% ~$ ~( Z
- {# @/ \8 x" s9 G) t/ {% l( q- b9 x
- key_state |= 0X2000; // 标记启动了700ms的超时定时0 L9 r. d: V. Q; h8 h( z: l h
- TIM4_Start(7000); // 启动700ms超时
3 v4 s& `) w8 Q; X1 E( R - }
* |, u0 x, q( W* C% ` - else // 按键处于松开状态3 p1 q+ }' R" b. r, r
- {
9 ~! b3 ~ W7 |0 I" { - if((key_state & 0X1FFF) == 1) // 按键按下计数=1,是单击事件
- u* h4 T1 ?+ V7 G- G - {( C% f! ]' ~ T( r& \( A* Y, y
- key_state = 0X8001;3 x# s' Y3 i9 p5 }2 T
- }else if((key_state & 0X1FFF) >= 2) // 按键按下计数>=2,是双击事件- J C! b$ ^+ g& Z" x
- {1 H) e4 O* K& b/ B6 Z0 ~1 h
- key_state = 0X8002;
! }; {3 Y) q% G' C - }else // 按键按下计数为其他值,本次按键事件无效
$ |6 x8 J. K/ h - {
J7 L" Z9 `: k4 j) U1 e - key_state = 0; V O/ G" ^7 Z5 D5 x- e
- }" Z/ c0 S0 d1 T% L
- }
$ W. i# n9 A8 Y& h6 C5 K( {0 h - }
7 C' L0 r/ a& W& m - }" ^2 F) A6 u3 x3 t
- }$ x: n( b3 l9 c1 z) C
- 1 X, j$ a2 y% s: n
- void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)) q& |; p. C6 Y( m" b/ k" Y
- {0 o, q, h) {2 f2 d* J% o! G# S# E
- switch(GPIO_Pin)
" F1 {4 a+ D/ C6 g9 H1 ~ - {2 u3 u) J! r& \6 S3 x5 T) H
- case KEY_Pin:4 J; N2 c: p+ K' v
- if((key_state & 0X8000) == 0)
' s. x3 m, ~( K3 C4 I3 \; A - {
7 D3 h/ D6 y' {5 n0 l7 v# x - key_state |= 0X4000; // 标记按键中断触发,在while死循环中进行消抖和的启动定时器* N* A! l" V" U) ?; _! u4 P0 c) H( S
- }
# a t$ j2 ?' R( ? - break;
* w& o9 j1 @3 i1 z8 m7 }- p - default:break;
8 ?7 c0 c9 K) h$ | Q - }
/ Z2 s; [. C+ d. b& p: v1 P+ F - }
/ b. ^2 b6 p# L. z. K
7 D' |* x! V: N3 x" i5 K. ?5 x# n- int main(void)
7 y# {( Y# b1 ]* k& y- m - {; }+ Q; d0 U- j$ i# r
- HAL_Init();
5 a% w5 T" s5 S3 C2 ~6 `) o3 H0 ~ - SystemClock_Config();
( R. D# R% j% Y% _ - MX_GPIO_Init();9 [1 O; D+ Z6 P: k$ h6 |3 X5 `
- MX_USART1_UART_Init();
7 O; c" m8 }" M; e5 S6 W6 t - MX_TIM4_Init();% O+ l& P9 P" g3 h, ~; d2 n& p" |7 v
- __HAL_TIM_CLEAR_IT(&htim4, TIM_IT_UPDATE); // STM32的定时器初始化完毕之后如果不清理中断标记为会有直接进入中断的问题
# ~7 F1 V/ i s4 M: O2 ? - 4 G8 _& h* _" a
- while (1)
0 @7 ^3 _2 F0 t8 W! t' T1 R - {0 D- y; f# {5 ?$ x) w0 K
- if(key_state & 0X4000) // 按键中断触发
+ A/ w( a$ |3 R0 p- Q7 u - {
6 `( |, C H- G! x1 w0 ? - key_state &= ~0x4000; // 清除事件
^! N7 _! a3 a9 \ - HAL_Delay(60); // 消抖延时
8 \$ ~3 Y8 a. q! J - 0 H9 ^. V! A$ N" f9 s
- // 消抖完毕之后再次检测按键是否为按下状态,如是则本次按键有效,否则视为无效并清除标识。1 u, u' W$ P( z! p* _
- if(HAL_GPIO_ReadPin(KEY_GPIO_Port, KEY_Pin) == GPIO_PIN_RESET)
: m) ]( K. r5 {3 s2 b+ T7 t' ~ - {
5 K. n+ k3 Y$ f+ Z' j - if((key_state & 0X1FFF) == 0) // 首次按下时启动300ms超时定时器9 H N) w# ~( k" q
- {
: _$ w# i5 Z6 Q' C: |3 T) A - TIM4_Start(3000);
( |* z! T! r; r N - } O4 [- v; n5 z
- key_state++; // 按键按下计数增加
9 }, [, P7 @" n+ | - }
9 @, b4 x. m, J# S - }
V0 o: Z/ O' c$ `$ {
$ d! G8 L4 U6 u2 D. b- if(key_state & 0X8000) // 按键事件发成了" T/ @1 w4 E& q, q- k
- {; { T' @/ `; r
- switch(key_state)) B5 v; }( i: f" D: C
- {+ c: B6 r3 {0 M! Q
- case 0X8001:7 `5 F* b2 v, ]2 r/ t' H* i- P4 k, `+ j
- printf("single\r\n");% \7 v" J" j8 r) u! `0 t6 v
- break;
4 j( y" l V# ]$ w - case 0X8002:; ` k2 Z' d; A# t5 m. U3 M! |/ t m
- printf("double\r\n");
3 A6 r. U! F" ]/ @ - break;
1 W o8 O' | { - case 0X8003:
2 \4 v* [. u: @4 {+ y - printf("hold\r\n");
# t; [3 R+ \. N/ G( m - break;
6 T c2 `6 Z& b2 n9 b - default:break;
1 w9 u! U- O) }: Z5 y0 {/ O0 b9 I' |# { - }
+ p; z, {$ B5 K# ^ - key_state = 0; // 清除事件* p- `' D% }8 y" ? Z
- }
9 a6 K3 U0 k0 P' @; c - }
# K* u5 r9 C7 ^ - }) v: a" n M: b& p+ Q, z
复制代码 ( O3 W: U) k( m) o% x
实验效果
+ _6 w" c& C9 M; x1 s按键单击、双击、长按实验效果如下:
5 {5 \. i) d$ c) G+ O- H; L
8 C! n4 o$ n. a# l
; A0 D: r. I' A
( k1 Z) u7 H$ c! S
* T+ _ t$ z1 R+ E( _/ j
|