1.电容触摸按键简介

image.png
原理:检测电容的充放电时间的方法来判断是否有触摸,图中 R 是外接的电容充
电电阻,Cs 是没有触摸按下时 TPAD 与 PCB 之间的杂散电容。而 Cx 则是有手指按下的时候,
手指与 TPAD 之间形成的电容图中的开关是电容放电开关(由实际使用时,由 STM32F4 的
IO 代替)。

先用开关将 Cs(或 Cs+Cx)上的电放尽,然后断开开关,让 R 给 Cs(或 Cs+Cx)充电,
当没有手指触摸的时候,Cs 的充电曲线如图中的 A 曲线。而当有手指触摸的时候,手指和 TPAD
之间引入了新的电容 Cx,此时 Cs+Cx 的充电曲线如图中的 B 曲线。从上图可以看出,A、B
两种情况下,Vc 达到 Vth 的时间分别为 Tcs 和 Tcs+Tcx。
其中,除了 Cs 和 Cx 我们需要计算,其他都是已知的,根据电容充放电公式:
Vc=V0*(1-e^(-t/RC))
其中 Vc 为电容电压,V0 为充电电压,R 为充电电阻,C 为电容容值,e 为自然底数,t 为充电时间。根据这个公式,我们就可以计算出 Cs 和 Cx。
当充点时间在 Tcs 附近,就可以认为没有触摸,而当充电时间大于 Tcs+Tx 时,就认为有触摸按下(Tx为检测阀值)。
**

2.原理图

image.png
用跳线帽短接多功能端口的 ADC 和 TPAD

3.源码

TPAD.C中
4 个比较重要的函数:TIM2_CH1_Cap_Init、TPAD_Get_Val、TPAD_Init 和 TPAD_Scan。
TIM2_CH1_Cap_Init函数,该函数和上一章的输入捕获函数基本一样,不同的是,这里我们设置的是 TIM2 上一章是 TIM5。通过该函数的设置,我们将可以捕获 PA5 上的上升沿,同样 TIM2 也是 32 位定时器。
TPAD_**Get_Val 函数,该函数用于得到定时器的一次捕获值。该函数先调用
TPAD**_Reset,将电容放电,同时设置通过调用函数TIM_SetCounter(TIM2,0)将计数值TIM2_CNT
设置为 0,然后死循环**等待发生上升沿捕获(或计数溢出),将捕获到的值(或溢出值)作为返
回值返回

**TPAD_Init 函数,该函数用于初始化输入捕获,并获取默认的 TPAD 值。该函数有一个参数,用来传递系统时钟,其实是为了配置 TIM2_CH1_Cap_Init 为 1us 计数周期。在该函数中连续 10 次读取 TPAD 值,将这些值升序排列后取中间 6 个值再做平均(这样做的目的是尽量减少误差),并赋值给 tpad_default_val,用于后续触摸判断的标准。

TPAD_Scan 函数,该函数用于扫描 TPAD 是否有触摸,该函数的参数mode,用于设置是否支持连续触发。返回值如果是 0,说明没有触摸,如果是 1,则说明有触摸。该函数同样包含了一个静态变量,用于检测控制,类似第八章的 KEY_Scan 函数。所以该函数同样是不可重入的。在函数中,我们通过连续读取 3 次(不支持连续按的时候)TPAD 的值,取这他们的最大值,和 tpad_default_val+TPAD_GATE_VAL 比较,如果大于则说明有触摸,如果小于,则说明无触摸。其中 tpad_default_val 是我们在调用 TPAD_Init 函数的时候得到的值,而 TPAD_GATE_VAL 则是我们设定的一个门限值(这个大家可以通过实验数据得出,根据实际情况选择适合的值就好了),这里我们设置为 100。

tpad.c

  1. #include "tpad.h"
  2. #include "delay.h"
  3. #include "usart.h"
  4. #define TPAD_ARR_MAX_VAL 0XFFFFFFFF //最大的ARR值(TIM2是32位定时器)
  5. vu16 tpad_default_val=0; //空载的时候(没有手按下),计数器需要的时间
  6. //初始化触摸按键
  7. //获得空载的时候触摸按键的取值.
  8. //psc:分频系数,越小,灵敏度越高.
  9. //返回值:0,初始化成功;1,初始化失败
  10. u8 TPAD_Init(u8 psc)
  11. {
  12. u16 buf[10];
  13. u16 temp;
  14. u8 j,i;
  15. TIM2_CH1_Cap_Init(TPAD_ARR_MAX_VAL,psc-1);//设置分频系数
  16. for(i=0;i<10;i++)//连续读取10次
  17. {
  18. buf[i]=TPAD_Get_Val();
  19. delay_ms(10);
  20. }
  21. for(i=0;i<9;i++)//排序
  22. {
  23. for(j=i+1;j<10;j++)
  24. {
  25. if(buf[i]>buf[j])//升序排列
  26. {
  27. temp=buf[i];
  28. buf[i]=buf[j];
  29. buf[j]=temp;
  30. }
  31. }
  32. }
  33. temp=0;
  34. for(i=2;i<8;i++)temp+=buf[i];//取中间的8个数据进行平均
  35. tpad_default_val=temp/6;
  36. printf("tpad_default_val:%d\r\n",tpad_default_val);
  37. if(tpad_default_val>TPAD_ARR_MAX_VAL/2)return 1;//初始化遇到超过TPAD_ARR_MAX_VAL/2的数值,不正常!
  38. return 0;
  39. }
  40. //复位一次
  41. //释放电容电量,并清除定时器的计数值
  42. void TPAD_Reset(void)
  43. {
  44. GPIO_InitTypeDef GPIO_InitStructure;
  45. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5; //PA5
  46. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT;//普通输出
  47. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz; //速度100MHz
  48. GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; //推挽
  49. GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_DOWN; //下拉
  50. GPIO_Init(GPIOA,&GPIO_InitStructure); //初始化PA5
  51. GPIO_ResetBits(GPIOA,GPIO_Pin_5);//输出0,放电
  52. delay_ms(5);
  53. TIM_ClearITPendingBit(TIM2, TIM_IT_CC1|TIM_IT_Update); //清除中断标志
  54. TIM_SetCounter(TIM2,0); //归0
  55. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5; //PA5
  56. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;//复用输出
  57. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;//速度100MHz
  58. GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; //推挽
  59. GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;//不带上下拉
  60. GPIO_Init(GPIOA,&GPIO_InitStructure); //初始化PA5
  61. }
  62. //得到定时器捕获值
  63. //如果超时,则直接返回定时器的计数值.
  64. //返回值:捕获值/计数值(超时的情况下返回)
  65. u16 TPAD_Get_Val(void)
  66. {
  67. TPAD_Reset();
  68. while(TIM_GetFlagStatus(TIM2, TIM_IT_CC1) == RESET)//等待捕获上升沿
  69. {
  70. if(TIM_GetCounter(TIM2)>TPAD_ARR_MAX_VAL-500)return TIM_GetCounter(TIM2);//超时了,直接返回CNT的值
  71. };
  72. return TIM_GetCapture1(TIM2);
  73. }
  74. //读取n次,取最大值
  75. //n:连续获取的次数
  76. //返回值:n次读数里面读到的最大读数值
  77. u16 TPAD_Get_MaxVal(u8 n)
  78. {
  79. u16 temp=0;
  80. u16 res=0;
  81. while(n--)
  82. {
  83. temp=TPAD_Get_Val();//得到一次值
  84. if(temp>res)res=temp;
  85. };
  86. return res;
  87. }
  88. //扫描触摸按键
  89. //mode:0,不支持连续触发(按下一次必须松开才能按下一次);1,支持连续触发(可以一直按下)
  90. //返回值:0,没有按下;1,有按下;
  91. #define TPAD_GATE_VAL 100 //触摸的门限值,也就是必须大于tpad_default_val+TPAD_GATE_VAL,才认为是有效触摸.
  92. u8 TPAD_Scan(u8 mode)
  93. {
  94. static u8 keyen=0; //0,可以开始检测;>0,还不能开始检测
  95. u8 res=0;
  96. u8 sample=3; //默认采样次数为3次
  97. u16 rval;
  98. if(mode)
  99. {
  100. sample=6; //支持连按的时候,设置采样次数为6次
  101. keyen=0; //支持连按
  102. }
  103. rval=TPAD_Get_MaxVal(sample);
  104. if(rval>(tpad_default_val+TPAD_GATE_VAL)&&rval<(10*tpad_default_val))//大于tpad_default_val+TPAD_GATE_VAL,且小于10倍tpad_default_val,则有效
  105. {
  106. if((keyen==0)&&(rval>(tpad_default_val+TPAD_GATE_VAL))) //大于tpad_default_val+TPAD_GATE_VAL,有效
  107. {
  108. res=1;
  109. }
  110. //printf("r:%d\r\n",rval);
  111. keyen=3; //至少要再过3次之后才能按键有效
  112. }
  113. if(keyen)keyen--;
  114. return res;
  115. }
  116. //定时器2通道2输入捕获配置
  117. //arr:自动重装值
  118. //psc:时钟预分频数
  119. void TIM2_CH1_Cap_Init(u32 arr,u16 psc)
  120. {
  121. GPIO_InitTypeDef GPIO_InitStructure;
  122. TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
  123. TIM_ICInitTypeDef TIM2_ICInitStructure;
  124. RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE); //TIM2时钟使能
  125. RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE); //使能PORTA时钟
  126. GPIO_PinAFConfig(GPIOA,GPIO_PinSource5,GPIO_AF_TIM2); //GPIOA5复用位定时器2
  127. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5; //GPIOA5
  128. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;//复用功能
  129. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz; //速度100MHz
  130. GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; //推挽复用输出
  131. GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;//不带上下拉
  132. GPIO_Init(GPIOA,&GPIO_InitStructure); //初始化PA5
  133. //初始化TIM2
  134. TIM_TimeBaseStructure.TIM_Period = arr; //设定计数器自动重装值
  135. TIM_TimeBaseStructure.TIM_Prescaler =psc; //预分频器
  136. TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; //设置时钟分割:TDTS = Tck_tim
  137. TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //TIM向上计数模式
  138. TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure); //根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位
  139. //初始化通道1
  140. TIM2_ICInitStructure.TIM_Channel = TIM_Channel_1; //CC1S=01 选择输入端 IC1映射到TIM2上
  141. TIM2_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Rising; //上升沿捕获
  142. TIM2_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI;
  143. TIM2_ICInitStructure.TIM_ICPrescaler = TIM_ICPSC_DIV1; //配置输入分频,不分频
  144. TIM2_ICInitStructure.TIM_ICFilter = 0x00;//IC2F=0000 配置输入滤波器 不滤波
  145. TIM_ICInit(TIM2, &TIM2_ICInitStructure);//初始化TIM2 IC1
  146. TIM_Cmd(TIM2,ENABLE ); //使能定时器2
  147. }

tapd.h

  1. #ifndef __TPAD_H_
  2. #define __TPAD_H_
  3. #include "stm32f4xx.h"
  4. #include "sys.h"
  5. //初始化触摸按键,获得空载的时候触摸按键的取值.
  6. //psc:分频系数,越小,灵敏度越高.
  7. //返回值:0,初始化成功;1,初始化失败
  8. u8 TPAD_Init(u8 psc);
  9. //复位一次
  10. //释放电容电量,并清除定时器的计数值
  11. void TPAD_Reset(void);
  12. //得到定时器捕获值
  13. //如果超时,则直接返回定时器的计数值.
  14. //返回值:捕获值/计数值(超时的情况下返回)
  15. u16 TPAD_Get_Val(void);
  16. //读取 n 次,取最大值
  17. //n:连续获取的次数
  18. //返回值:n 次读数里面读到的最大读数值
  19. u16 TPAD_Get_MaxVal(u8 n);
  20. //扫描触摸按键
  21. //mode:0,不支持连续触发(按下一次必须松开才能按下一次);
  22. // 1,支持连续触发(可以一直按下)
  23. //返回值:0,没有按下;1,有按下;
  24. u8 TPAD_Scan(u8 mode);
  25. //定时器 2 通道 2 输入捕获配置
  26. //arr:自动重装值
  27. //psc:时钟预分频数
  28. void TIM2_CH1_Cap_Init(u32 arr,u16 psc);
  29. #endif

led.h

利用位带操作
#define LED0 PFout(9) // DS0
#define LED1 PFout(10) // DS1

main.c

  1. #include "led.h" // 将接口驱动头文件包含进来
  2. #include "tpad.h"
  3. #include "usart.h"
  4. #include "delay.h"
  5. int main(void)
  6. {
  7. u8 t=0;
  8. NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置系统中断优先级分组 2
  9. delay_init(168); //初始化延时函数
  10. uart_init(115200); //初始化串口波特率为 115200
  11. LED_Init(); //初始化 LED
  12. TPAD_Init(8); //初始化触摸按键,以 84/4=21Mhz 频率计数
  13. while(1)
  14. {
  15. if(TPAD_Scan(0)) //成功捕获到了一次上升沿(此函数执行时间至少 15ms)
  16. {
  17. LED1=!LED1; //LED1 取反
  18. }
  19. t++;
  20. if(t==15)
  21. {
  22. t=0; LED0=!LED0; //LED0 取反,提示程序正在运行
  23. }
  24. delay_ms(10);
  25. }
  26. }