当前位置: 首页>編程日記>正文

stm32_电容触摸按键

stm32_电容触摸按键

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

  • 先用开关将 Cs(或 Cs+Cx)上的电放尽,然后断开开关,让 R 给 Cs(或 Cs+Cx)充电;
  • 当没有手指触摸的时候,Cs 的充电曲线如图中的 A 曲线。而当有手指触摸的时候,手指和 TPAD之间引入了新的电容 Cx,此时 Cs+Cx 的充电曲线如图中的 B 曲线。电容越大,充电一定时,时间越多
  • 电容充放电公式: :Vc=V0*(1-e^(-t/RC))
    实现触摸检测:当充电时间在 Tcs 附近,就可以认为没有触摸,而当充电时间大于 Tcs+Tx 时,就认为有触摸按下(Tx为检测阀值)
    在 MCU 每次复位重启的时候,我们执行一次捕获检测(没触摸),记录此时的值为 tpad_default_val,作为判断的依据。

    检测电容触摸按键过程:
    ①TPAD引脚设置为推挽输出,输出0,实现电容放电到0
    ②TPAD引脚设置为浮空输入(IO复位后的状态),电容开始充电
    ③同时开启TPAD引脚的输入捕获开始捕获
    ④等待充电完成(充电到底为Vx,检测到上升沿)
    ⑤计算充电时间:在 MCU 每次复位重启的时候,我们执行一次捕获检测(没触摸),记录此时的值,记为 tpad_default_val,作为判断的依据。
#include "tpad.h"
#include "stm32f4xx.h"
#include "delay.h"#define TPAD_ARR_MAX_VAL  0XFFFFFFFF	//最大的ARR值(TIM2是32位定时器)	  //定时器2通道1输入捕获
//arr为自动重装载值,psc为预分频数void TIM2_CH1_CAP_Init(u8 psc)
{GPIO_InitTypeDef  GPIO_InitStructure; TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;TIM_ICInitTypeDef  TIM2_ICInitStructure;//使能时钟RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE);  	  //TIM2时钟使能    RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE); 	//使能PORTA时钟	//初始化IO口,复用功能GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;//复用功能GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;	//速度100MHzGPIO_InitStructure.GPIO_OType = GPIO_OType_PP; //推挽复用输出GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;//不带上下拉  GPIO_Init(GPIOA,&GPIO_InitStructure);//设置引脚复用映射GPIO_PinAFConfig(GPIOA,GPIO_PinSource5,GPIO_AF_TIM2); //GPIOA5复用位定时器2//初始化定时器ARR,PSCTIM_TimeBaseStructure.TIM_Period = arr; //设定计数器自动重装值   TIM_TimeBaseStructure.TIM_Prescaler =psc; 	//预分频器 	   TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; //设置时钟分割:TDTS = Tck_timTIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  //TIM向上计数模式TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure); //根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位//初始化定时器2,输入捕获通道TIM_TimeBaseStructure.TIM_Period = arr;  //设定计数器自动重装值   TIM_TimeBaseStructure.TIM_Prescaler = psc; //预分频器 	TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  //TIM向上计数模式TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; //设置时钟分割:TDTS = Tck_timTIM_TimeBaseStructure.TIM_RepetitionCounter = 0; //默认设置设为0,可不设置TIM_TimeBaseInit(TIM2,&TIM_TimeBaseStructure);   //初始化TIM2 TIM2_ICInitStructure.TIM_Channel = TIM_Channel_1;   //CC1S=01 	选择输入端 TIM2的通道1TIM2_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Rising;	//选择捕获极性,上升沿捕获TIM2_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI;  //映射自身的TI,不映射其它通道的TITIM2_ICInitStructure.TIM_ICPrescaler = TIM_ICPSC_DIV1;	 //配置输入分频,不分频 TIM2_ICInitStructure.TIM_ICFilter = 0x00;             //IC2F=0000 配置输入滤波器 不滤波TIM_ICInit(TIM2,&TIM2_ICInitStructure);          //初始化通道1/*开启捕获中断,设置中断优先级(视情况而定)*/	//使能定时器TIM_Cmd(TIM2,ENABLE ); 	//使能定时器2}vu16 tpad_default_val=0;				//空载的时候(没有手按下),计数器需要的时间//获得空载的时候触摸按键的取值,即无触摸时的取值
//psc:分频系数,越小,灵敏度越高.
//返回值:0,初始化成功;1,初始化失败
u8 TPAD_Init(u8 psc)
{u16 buf[10];u16 temp;u8 j,i;TIM2_CH1_Cap_Init(TPAD_ARR_MAX_VAL,psc-1);//设置分频系数for(i=0;i<10;i++)//连续读取10次{				 buf[i]=TPAD_Get_Val();                //捕获到上升沿时返回的寄存器的值delay_ms(10);	    }				    for(i=0;i<9;i++)			//对捕获到的10个值进行排序{for(j=i+1;j<10;j++){if(buf[i]>buf[j])  //升序排列{temp=buf[i];buf[i]=buf[j];buf[j]=temp;}}}temp=0;for(i=2;i<8;i++)temp+=buf[i];      //取中间的8个数据进行平均tpad_default_val=temp/6;printf("tpad_default_val:%d\r\n",tpad_default_val);	if(tpad_default_val>TPAD_ARR_MAX_VAL/2)return 1;      //初始化遇到超过TPAD_ARR_MAX_VAL/2的数值,不正常!return 0;		     	    					   
}//复位一次
//释放电容电量,并清除定时器的计数值,一般作为第一步
//等待捕获void TPAD_Reset(void)
{		GPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;  //PA5GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT;//普通输出GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;	//速度100MHzGPIO_InitStructure.GPIO_OType = GPIO_OType_PP; //推挽 GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_DOWN; //下拉GPIO_Init(GPIOA,&GPIO_InitStructure); //初始化PA5GPIO_ResetBits(GPIOA,GPIO_Pin_5);//输出0,放电delay_ms(5);TIM_ClearITPendingBit(TIM2, TIM_IT_CC1|TIM_IT_Update); //清除中断标志TIM_SetCounter(TIM2,0);		//归0GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5; //PA5GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;//复用输出GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;//速度100MHzGPIO_InitStructure.GPIO_OType = GPIO_OType_PP; //推挽 GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;//不带上下拉 GPIO_Init(GPIOA,&GPIO_InitStructure); //初始化PA5}//得到定时器捕获值
//如果超时,则直接返回定时器的计数值.
//返回值:捕获值/计数值(超时的情况下返回)u16 TPAD_Get_Val(void)
{				   TPAD_Reset();while(TIM_GetFlagStatus(TIM2, TIM_IT_CC1) == RESET)//等待捕获上升沿{if(TIM_GetCounter(TIM2)>TPAD_ARR_MAX_VAL-500)return TIM_GetCounter(TIM2);//超时了,直接返回CNT的值?为什么-500?};	return TIM_GetCapture1(TIM2);	 
} 	  //读取n次,取最大值
//n:连续获取的次数
//返回值:n次读数里面读到的最大读数值
u16 TPAD_Get_MaxVal(u8 n)
{u16 temp=0;u16 res=0; while(n--){temp=TPAD_Get_Val();//得到一次值if(temp>res)res=temp;};return res;
}  //扫描触摸按键
//mode:0,不支持连续触发(按下一次必须松开才能按下一次);1,支持连续触发(可以一直按下)
//返回值:0,没有按下;1,有按下;										  
#define TPAD_GATE_VAL 	100	//触摸的门限值(一个大概值),也就是必须大于tpad_default_val+TPAD_GATE_VAL,才认为是有效触摸.
u8 TPAD_Scan(u8 mode)
{static u8 keyen=0;	//0,可以开始检测;>0,还不能开始检测	 u8 res=0;u8 sample=3;		//默认采样次数为3次	 u16 rval;if(mode){sample=6;	//支持连按的时候,设置采样次数为6次keyen=0;	//支持连按	  }rval=TPAD_Get_MaxVal(sample); if(rval>(tpad_default_val+TPAD_GATE_VAL)&&rval<(10*tpad_default_val))//大于tpad_default_val+TPAD_GATE_VAL,且小于10倍tpad_default_val,则有效{							 if((keyen==0)&&(rval>(tpad_default_val+TPAD_GATE_VAL)))	//大于tpad_default_val+TPAD_GATE_VAL,有效{res=1;}	   //printf("r:%d\r\n",rval);		     	    					   keyen=3;				//至少要再过3次之后才能按键有效   } if(keyen)keyen--;		   							   		     	    					   return res;
}	 


https://www.fengoutiyan.com/post/15308.html

相关文章:

  • stm32按键输入
  • stm32供电电路
  • STM32电压表
  • 单键触摸芯片
  • 电容式触摸按键工作原理
  • stm32测量电压
  • 触摸按键电路
  • 电容触摸ic
  • 鏡像模式如何設置在哪,圖片鏡像操作
  • 什么軟件可以把圖片鏡像翻轉,C#圖片處理 解決左右鏡像相反(旋轉圖片)
  • 手機照片鏡像翻轉,C#圖像鏡像
  • 視頻鏡像翻轉軟件,python圖片鏡像翻轉_python中鏡像實現方法
  • 什么軟件可以把圖片鏡像翻轉,利用PS實現圖片的鏡像處理
  • 照片鏡像翻轉app,java實現圖片鏡像翻轉
  • 什么軟件可以把圖片鏡像翻轉,python圖片鏡像翻轉_python圖像處理之鏡像實現方法
  • matlab下載,matlab如何鏡像處理圖片,matlab實現圖像鏡像
  • 圖片鏡像翻轉,MATLAB:鏡像圖片
  • 鏡像翻轉圖片的軟件,圖像處理:實現圖片鏡像(基于python)
  • canvas可畫,JavaScript - canvas - 鏡像圖片
  • 圖片鏡像翻轉,UGUI優化:使用鏡像圖片
  • Codeforces,CodeForces 1253C
  • MySQL下載安裝,Mysql ERROR: 1253 解決方法
  • 勝利大逃亡英雄逃亡方案,HDU - 1253 勝利大逃亡 BFS
  • 大一c語言期末考試試題及答案匯總,電大計算機C語言1253,1253《C語言程序設計》電大期末精彩試題及其問題詳解
  • lu求解線性方程組,P1253 [yLOI2018] 扶蘇的問題 (線段樹)
  • c語言程序設計基礎題庫,1253號C語言程序設計試題,2016年1月試卷號1253C語言程序設計A.pdf
  • 信奧賽一本通官網,【信奧賽一本通】1253:抓住那頭牛(詳細代碼)
  • c語言程序設計1253,1253c語言程序設計a(2010年1月)
  • 勝利大逃亡英雄逃亡方案,BFS——1253 勝利大逃亡
  • 直流電壓測量模塊,IM1253B交直流電能計量模塊(艾銳達光電)
  • c語言程序設計第三版課后答案,【渝粵題庫】國家開放大學2021春1253C語言程序設計答案
  • 18轉換為二進制,1253. 將數字轉換為16進制
  • light-emitting diode,LightOJ-1253 Misere Nim
  • masterroyale魔改版,1253 Dungeon Master
  • codeformer官網中文版,codeforces.1253 B
  • c語言程序設計考研真題及答案,2020C語言程序設計1253,1253計算機科學與技術專業C語言程序設計A科目2020年09月國家開 放大學(中央廣播電視大學)
  • c語言程序設計基礎題庫,1253本科2016c語言程序設計試題,1253電大《C語言程序設計A》試題和答案200901
  • 肇事逃逸車輛無法聯系到車主怎么辦,1253尋找肇事司機