本文主要是介绍STM32 CortexM4 主控板触摸屏保姆级别总结,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
一、实物图
二、触摸屏种类分类
电阻式屏幕
特点:定位准确且可以实现单点触摸;电阻式屏幕是一种传感器,它将矩形区域的触摸点的物理位置转换为代表X/Y轴的电压,现在很多的LCD屏幕都是采用了电阻式的触摸屏,这种屏幕可以用5、6、7、8线来产生偏置电压,同时可以读回触摸点的电压。
电阻式触摸屏是一种传感器,基本上是薄膜加上玻璃的结构,薄膜和玻璃相邻的一面上均涂有ITO(纳米铟锡金属氧化物)涂层,ITO具有很好的导电性和透明性。当触摸操作时,薄膜下层的ITO会接触到玻璃上层的ITO,经由感应器传出相应的电信号,经过转换电路送到处理器,通过运算转化为屏幕上的X、Y值,而完成点选的动作,并呈现在屏幕上。
其优缺点也很明显,缺点是他只能支持单点触摸,不能实现多点触发触摸功能;但是由于这个单点触摸的特点,使得它触摸的精度很高。
电感式触摸屏
支持多点触摸,价格偏贵。工业应用最广泛,我们现在市面上的手机都是感应式的触摸屏,它的优缺点是跟电阻式的触摸屏相反,优点是可以实现多点感应,而缺点就是精度没那么高。
红外线式
特点是:价格十分低廉,但其外框易碎,容易产生光干扰,曲面情况下失真,所以一般情况下我们不会选择这样的触摸屏。这个一般用在特殊的工艺。
表面声波式
特点是:可以解决各种缺点,但是屏幕表面如果有水滴和尘土会使触摸屏变的迟钝。
XPT2046采集触摸AD芯片
触摸屏接口连接原理
XPT2046特性
工作电压范围为2.2V-5.25V,支持1.5V~5.25V的数字IO口,内建2.5V参考电压源;
电源电压测量(0V-6V)内建结温测量功能触摸压力测量;
采用SPI3线控制通信接口且有自动power-down功能;
封装:QPN-16.TSSOP-16和VFBGA-48与TSC2046、AK4182A完全兼容
XPT2046结构原理
XPT2046引脚
屏幕X/Y坐标的输入+/-端
应用引脚
使用到的引脚
看原理图,我们知道,有红框圈起来的引脚要用我们进行配置:
T_MISO PB2 引脚
XPT2046的串行数据输出端。STM32的数据输入端,数据在DCLK的下降沿 移出,当CS高电平时为高阻状态,配置为输入模式,空闲为高。
T_PEN PB1引脚
笔中断输出,配置为输入模式,空闲模式为高电平;
T_CS PC13引脚
片选信号。控制转换时序和使能串行输入输出寄存器,高电平时ADC掉电;
T_MOSI PF11引脚
XPT2046的串行数据输入端,STM32的输出端,当CS为低电平时,数据在DCLK上升沿锁存进来;
T_SCK PB0引脚
XPT2046外部时钟信号输入,STM32的时钟信号输出;
小结
STM32发数据的时候,片选拉低,DCLK拉高,把数据锁存在T_MOSI引脚传输出去;当STM32读数据的时候,片选拉低,DCLK拉低,读取T_MISO电平状态。
XPT2046控制说明
单端还是差分模式
选择VBAT、 Temp和AUX时可以配置为单端模式,作为触摸屏应用时,可以配置为差分模式,这可有效消除由于驱动开关的寄生电阻及外部的干扰带来的测量误差,提高转换准确
度。
笔中断输出
有人按下屏幕时,触摸屏下拉到地,也就是按下为低电平
控制字节
用于选择采集的是X还是Y轴的AD值,作用:控制字节由 DIN 输入的控制字如表 3 所示,它用来启动转换,寻址,设置 ADC 分辨率,配置和对 XPT2046 进行掉电控
制。
代码实例
扫描触摸屏
touch:存放读取到的x/y坐标值,返 回 值 : 0:有按下;1:无按下
********************************************************************************/
unsigned char Xpt2046_ScanTouch(Touch_Typedef *touch)
{Touch_Typedef ad;unsigned char ucRetVaule;if(T_PEN == 0){ //判断触摸屏有没有被按下if(T_PEN == 0){Xpt2046_ReadXYAD(&ad);touch->xval = ad.xval * Kx + Bx;touch->yval = ad.yval * Ky + By;ucRetVaule = 0;}}else{touch->xval=0xffff;touch->yval=0xffff;ucRetVaule = 1;}return ucRetVaule;
}
读取X/Y轴AD值,并滤波处理
void Xpt2046_ReadXYAD(Touch_Typedef *touch)
{unsigned char i, j;unsigned short adx[5], ady[5];unsigned short temp;for(i=0; i<5; i++){ //先采集多次数据adx[i] = Xpt2046_ReadAD(0xd0);ady[i] = Xpt2046_ReadAD(0x90);}for(i=0; i<5; i++){ //先排序,找出最大和最小的值for(j=0; j<5-i-1; j++){if(adx[j] > adx[j+1]){temp = adx[j];adx[j] = adx[j+1];adx[j+1] = temp;}if(ady[j] > ady[j+1]){temp = ady[j];ady[j] = ady[j+1];ady[j+1] = temp;}}}touch->xval = adx[2];touch->yval = ady[2];
}
读取轴AD值
unsigned short Xpt2046_ReadAD(unsigned char cmd)
{unsigned char vh, vl;T_CS = 0; //拉低片选,选中器件,开始通信Xpt2046_WriteByte(cmd); //发送测量命令Delay_Us(1); //等到芯片忙结束vh = Xpt2046_ReadByte();//读取数据的高位vl = Xpt2046_ReadByte();//读取数据的低位T_CS = 1; //拉低片选,取消选中,结束通信return ((vh<<8|vl)>>4); //返回结果,其中只有12位有效
}
触摸屏校准函数
这个可以通用,需要的直接复制就好了
void Xpt2046_TouchAdjust(void)
{unsigned short lcd_pos[4][2] = {20, 20, 300, 20, 20, 460, 300, 460, };Touch_Typedef touch_pos[4]; //用来存放x,y的4个AD值unsigned char i, j;double len1 = 0.00f, len2 = 0.00f;Lcd_ShowString(30, 130, "touch adjust start", RED, WHITE, 16);while(1){for(i=0; i<4; i++){ //读取4个对应的触摸屏坐标for(j=0; j<30; j++){ //画一个十字架Lcd_DrawPoint(lcd_pos[i][0]-15+j, lcd_pos[i][1], RED);Lcd_DrawPoint(lcd_pos[i][0], lcd_pos[i][1]-15+j, RED);}printf("等待校验\r\n");while(T_PEN == 1); //等待按下触摸屏 Delay_Ms(50); //延时50ms待数据稳定printf("按下触摸屏\r\n");Xpt2046_ReadXYAD(&touch_pos[i]);//获得触摸屏测量的x,y轴数值while(T_PEN == 0); //等待松开手Delay_Ms(200);for(j=0; j<30; j++){ //清掉十字架图标Lcd_DrawPoint(lcd_pos[i][0]-15+j, lcd_pos[i][1], WHITE);Lcd_DrawPoint(lcd_pos[i][0], lcd_pos[i][1]-15+j, WHITE);}}//校验坐标-计算点击的触摸点是否正确 如果不正确重新校准//水平两个点之间的距离比较 len1 = (float)sqrt((touch_pos[1].xval-touch_pos[0].xval)*(touch_pos[1].xval-touch_pos[0].xval) \+ (touch_pos[1].yval-touch_pos[0].yval)*(touch_pos[1].yval-touch_pos[0].yval));len2 = (float) sqrt((touch_pos[3].xval-touch_pos[2].xval)*(touch_pos[3].xval-touch_pos[2].xval) \+ (touch_pos[3].yval-touch_pos[2].yval)*(touch_pos[3].yval-touch_pos[2].yval)); if(((len1/len2)<0.95) || ((len1/len2)>1.05)){continue; }//垂直两个点之间的距离比较len1 = (float)sqrt((touch_pos[2].xval-touch_pos[0].xval)*(touch_pos[2].xval-touch_pos[0].xval) \+ (touch_pos[2].yval-touch_pos[0].yval)*(touch_pos[2].yval-touch_pos[0].yval));len2 = (float)sqrt((touch_pos[3].xval-touch_pos[1].xval)*(touch_pos[3].xval-touch_pos[1].xval) \+ (touch_pos[3].yval-touch_pos[1].yval)*(touch_pos[3].yval-touch_pos[1].yval)); if(((len1/len2)<0.95) || ((len1/len2)>1.05)){continue; //点击的点不符合要求} //对角线两个点之间的距离比较len1 = (float)sqrt((touch_pos[3].xval-touch_pos[0].xval)*(touch_pos[3].xval-touch_pos[0].xval) \+ (touch_pos[3].yval-touch_pos[0].yval)*(touch_pos[3].yval-touch_pos[0].yval));len2 = (float)sqrt((touch_pos[2].xval-touch_pos[1].xval)*(touch_pos[2].xval-touch_pos[1].xval) \+ (touch_pos[2].yval-touch_pos[1].yval)*(touch_pos[2].yval-touch_pos[1].yval)); if(((len1/len2)<0.95) || ((len1/len2)>1.05)){continue;} //计算校准参数 Kx (Ky)--斜率;Bx(By) --偏移量//计算x映射 Xlcd = Kx * touch_x + BxKx = (float)(lcd_pos[1][0]-lcd_pos[0][0]) / (touch_pos[1].xval-touch_pos[0].xval);Bx = lcd_pos[0][0] - Kx*touch_pos[0].xval;//计算y映射 Ylcd = Ky*touch_y + ByKy = (float)(lcd_pos[2][1]-lcd_pos[0][1]) / (touch_pos[2].yval-touch_pos[0].yval);By = lcd_pos[0][1] - Ky*touch_pos[0].yval; Lcd_Fill(0, 0, 320, 480, WHITE);Lcd_ShowString(30, 130, "touch adjust OK", RED, WHITE, 16);printf("校准参数 Ky=%f;Kx=%f;By=%d;Bx=%d;\r\n",Ky, Kx, By, Bx);Delay_Ms(1000);Delay_Ms(1000);Lcd_Fill(0, 0, 320, 480, WHITE);break; }
}
这篇关于STM32 CortexM4 主控板触摸屏保姆级别总结的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!