要用到触摸屏首先就要对触摸屏的原理有一定的了解,我想这个是前提,也不用太多说的。就是当触笔触到屏上时,对应的位置就会产生相应大小的电压,输入到芯片,AD转换后得到一个数据。而触摸校准就是将接受到的原始模数转换值转换成屏幕像素坐标。

cubemx配置stm32触摸屏 stm32触摸屏控制_cubemx配置stm32触摸屏

再就是了解触摸芯片,知道他的工作方式,以及跟STM32的连线。触摸实验中,我的实验板是用SPI口来实现数据的传输的,即SPI与xpt2046相连。

cubemx配置stm32触摸屏 stm32触摸屏控制_cubemx配置stm32触摸屏_02

触摸屏控制芯片ADS7843中文资料

  ADS7843是一个内置12位模数转换、低导通电阻模拟开关的串行接口芯片。供电电压2.7~5 V,参考电压VREF为1 V~+VCC,转换电压的输入范围为0~ VREF,最高转换速率为125 kHz。

ADS7843引脚图及引脚功能说明了:

 

cubemx配置stm32触摸屏 stm32触摸屏控制_数据_03

ADS7843的引脚配置如图3所示。表1为引脚功能说明,图4为典型应用。

 

cubemx配置stm32触摸屏 stm32触摸屏控制_引脚_04

ADS7843典型应用电路

cubemx配置stm32触摸屏 stm32触摸屏控制_cubemx配置stm32触摸屏_05

ADS7843之所以能实现对触摸屏的控制,是因为其内部结构很容易实现电极电压的切换,并能进行快速A/D转换。图5所示为其内部结构,A2~A0和SER/为控制寄存器中的控制位,用来进行开关切换和参考电压的选择。

ADS7843之所以能实现对触摸屏的控制,是因为其内部结构很容易实现电极电压的切换,并能进行快速A/D转换。图5所示为其内部结构,A2~A0和SER/为控制寄存器中的控制位,用来进行开关切换和参考电压的选择。

 

我们传统的鼠标是一种相对定位系统,只和前一次鼠标的位置坐标有关。而触摸屏则是一种绝对坐标系统,要选哪就直接点哪,与相对定位系统有着本质的区别。绝对坐标系统的特点是每一次定位坐标与上一次定位坐标没有关系,每次触摸的数据通过校准转为屏幕上的坐标,不管在什么情况下,触摸屏这套坐标在同一点的输出数据是稳定的。不过由于技术原理的原因,并不能保证同一点触摸每一次采样数据相同,不能保证绝对坐标定位,点不准,这就是触摸屏最怕出现的问题:漂移。对于性能质量好的触摸屏来说,漂移的情况出现并不是很严重。

所以很多应用触摸屏的系统启动后,进入应用程序前,先要执行校准程序。  通常应用程序中使用的LCD 坐标是以像素为单位的。比如说:左上角的坐标是一组非 0 的数值,比如(20,20),而右下角的坐标为(220,300)。这些点的坐标都是以像素为单位的,而从触摸屏中读出的是点的物理坐标,其坐标轴的方向、XY 值的比例因子、偏移量都与 LCD 坐标不同,所以,可以在程序中使用一个函数(我们采用 Convert_Pos 函数)中把物理坐标首先转换为像素坐标,然后再赋给 POS 结构,达到坐标转换的目的。

校正思路:在了解了校正原理之后,我们可以得出下面的一个从物理坐标到像素坐标的转换关系式:

                                                LCDx=xfac*Px+xoff;

                                                LCDy=yfac*Py+yoff;

其中(LCDx,LCDy)是在 LCD 上的像素坐标,(Px,Py)是从触摸屏读到的物理坐标。xfac,yfac 分别是 X 轴方向和 Y 轴方向的比例因子,而 xoff 和 yoff 则是这两个方向的偏移量。

这样我们只要事先在屏幕上面显示 4 个点(这四个点的坐标是已知的),分别按这四个点就可以从触摸屏读到 4 个物理坐标,这样就可以通过待定系数法求出 xfac、yfac、xoff、yoff 这四个参数。我们保存好这四个参数,在以后的使用中,我们把所有得到的物理坐标都按照这个关系式来计算,得到的就是准确的屏幕坐标。达到了触摸屏校准的目的。

以下是自己的校准程序(在正点原子上做了一些改变,没有用到中断):

//触摸屏校准代码
 //得到四个校准参数
void Touch_Adjust(void)
 {        
  u16 pos_temp[4][2];//坐标缓存值
  u8  cnt=0; 
  u16 d1,d2;
  u32 tem1,tem2;
  float fac;    
  cnt=0;    
  TextColor=Blue;
  BackColor =White;      //TextColor = 0x0000, BackColor = 0xFFFF;
  LCD_Clear(White);//清屏  
 // POINT_COLOR=RED;//红色
 // LCD_Clear(WHITE);//清屏
  Drow_Touch_Point(20,20);//画点1
  delay_ms(1000);
 // Pen_Point.Key_Sta=Key_Up;//消除触发信号
 // Pen_Point.xfac=0;//xfac用来标记是否校准过,所以校准之前必须清掉!以免错误 
  while(1)
  {
   if(Read_ADS2(&x,&y))//按键按下了
   {
    //if(Read_TP_Once())//得到单次按键值
   // {            
     pos_temp[cnt][0]=x;
     pos_temp[cnt][1]=y;
     cnt++;
     delay_ms(100);
   // }   
    switch(cnt)
    {     
     case 1:
      LCD_Clear(White);//清屏
      delay_ms(1000);
      Drow_Touch_Point(20,300);//画点2
      break;
     case 2:
      LCD_Clear(White);//清屏
      delay_ms(1000);
      Drow_Touch_Point(220,20);//画点3
      break;
     case 3:
      LCD_Clear(White);//清屏
      delay_ms(1000);
      Drow_Touch_Point(220,300);//画点4
      break;
     case 4:  //全部四个点已经得到
            //对边相等
      tem1=abs(pos_temp[0][0]-pos_temp[1][0]);//x1-x2
      tem2=abs(pos_temp[0][1]-pos_temp[1][1]);//y1-y2
      tem1*=tem1;
      tem2*=tem2;
      d1=sqrt(tem1+tem2);//得到1,2的距离
      
      tem1=abs(pos_temp[2][0]-pos_temp[3][0]);//x3-x4
      tem2=abs(pos_temp[2][1]-pos_temp[3][1]);//y3-y4
      tem1*=tem1;
      tem2*=tem2;
      d2=sqrt(tem1+tem2);//得到3,4的距离
      fac=(float)d1/d2;
      if(fac<0.95||fac>1.05||d1==0||d2==0)//不合格
      {
       cnt=0;
       LCD_Clear(White);//清屏
       Drow_Touch_Point(20,20);
       continue;
      }
      tem1=abs(pos_temp[0][0]-pos_temp[2][0]);//x1-x3
      tem2=abs(pos_temp[0][1]-pos_temp[2][1]);//y1-y3
      tem1*=tem1;
      tem2*=tem2;
      d1=sqrt(tem1+tem2);//得到1,3的距离
      
      tem1=abs(pos_temp[1][0]-pos_temp[3][0]);//x2-x4
      tem2=abs(pos_temp[1][1]-pos_temp[3][1]);//y2-y4
      tem1*=tem1;
      tem2*=tem2;
      d2=sqrt(tem1+tem2);//得到2,4的距离
      fac=(float)d1/d2;
      if(fac<0.95||fac>1.05)//不合格
      {
       cnt=0;
       LCD_Clear(White);//清屏
       Drow_Touch_Point(20,20);
       continue;
      }//正确了
           
      //对角线相等
      tem1=abs(pos_temp[1][0]-pos_temp[2][0]);//x1-x3
      tem2=abs(pos_temp[1][1]-pos_temp[2][1]);//y1-y3
      tem1*=tem1;
      tem2*=tem2;
      d1=sqrt(tem1+tem2);//得到1,4的距离
  
      tem1=abs(pos_temp[0][0]-pos_temp[3][0]);//x2-x4
      tem2=abs(pos_temp[0][1]-pos_temp[3][1]);//y2-y4
      tem1*=tem1;
      tem2*=tem2;
      d2=sqrt(tem1+tem2);//得到2,3的距离
      fac=(float)d1/d2;
      if(fac<0.95||fac>1.05)//不合格
      {
       cnt=0;
       LCD_Clear(White);//清屏
       Drow_Touch_Point(20,20);
       continue;
      }//正确了
      //计算结果
      x1=pos_temp[0][0];
      x2=pos_temp[1][0];
      x3=pos_temp[2][0];
      x4=pos_temp[3][0];
      y1=pos_temp[0][1];
      y2=pos_temp[1][1];
      y3=pos_temp[2][1];
      y4=pos_temp[3][1];
      TextColor=Blue;
      LCD_Clear(White);//清屏
      ili9320_PutChars(120,120,"Touch Screen Adjust OK!",23,TextColor,BackColor);//校正完成
      delay_ms(1000);
      LCD_Clear(Blue);//清屏  
      return;//校正完成    
    }
   }
  }
 } 
我的算法,跟正点原子不同,他的我没有理解。
以下是带清屏的程序:
 while (1)
     {
   if(Read_ADS2(&X,&Y))
      {
    zuobiaox=20+(s32)(X-x1)*200/(x3-x1);
      zuobiaoy=20+(s32)(Y-y1)*280/(y2-y1);
    if(zuobiaox>0&&zuobiaox<20&&zuobiaoy>0&&zuobiaoy<40)
    {
     LCD_Clear(Blue);//清屏
     LCD_DrawRectFill(0, 0, 20, 40,Yellow);
     ili9320_PutChars(0,0,"RST",3,Red,Yellow); 
    }          
    else if(zuobiaox>0&&zuobiaox<240&&zuobiaoy>0&&zuobiaoy<320)
      lcd_fill_circle(zuobiaox,zuobiaoy, Red);
      }
  }
----------------------
     |  +  (x1,y1)    + | (x2,y2)
     |                    |
     |  +   (x3,y3)     + | (x4,y4)
     ----------------------



触摸屏校准

上图理解为触摸屏,应为实际应用中触摸屏边角出电压相对不稳定,所以放弃一部分,四个点分别为(20,20)(20,300)(220,20)(220,300),对应的AD值分别设为(x1,y1)(x2,y2)(x3,y3)(x4,y4).
      首先:在LCD上绘制四个对应的“+”用来取较准点;
其次:点四个“+”,得到各点的AD值,由于其中的电阻是线性的,可以根据x轴上两点的x的AD值如x1,x3以及对应LCD坐标值求出x轴上x的AD值与LCD坐标值对应的关系,是一条直线的方程,再根据此方程由AD值求LCD坐标值即可;y轴上也是一样的道理。
      最后:计算  (注:触摸屏电压最小点,为LCD显示坐标最大点,(X,Y)为触摸电压)

zuobiaox=20+(s32)(X-x1)*200/(x3-x1);

zuobiaoy=20+(s32)(Y-y1)*280/(y2-y1);

-------------------------

→_→在校准函数中,不断扫描TP_Scan()函数,如果这时候你触摸了一下屏幕,PEN所对应STM32的引脚将会从高电平跳变为低电平,详情看上文第二讲的注意→_→。即Ttp_dev.sta=1100,0000(根据上图第一个方框得出)。不满足校准函数中的if((tp_dev.sta&0xc0)==TP_CATH_PRES),故不会进行下面的画点。如果之前并没有按下触摸屏,这时同样是不满足上面if的。如果之前按下后松下了,这时Ttp_dev.sta=0111,1111,这时满足校准函数中的if((tp_dev.sta&0xc0)==TP_CATH_PRES),然后在校准函数中标记下触摸已经被处理了(清除tp_dev.sta),清除第一个点,画第二个点,清除第二个点,画第三个点,清除第三个点,画第四个点,清除第四个点。也就是,触摸屏幕有两个状态:按下和松开。当按下时,程序执行的是将按下的AD值坐标存到两个数组中即上图中的TP_Read_XY2(&tp_dev.x[0],&tp_dev.y[0]);当松开时,清除原来的点,并画一个新点。这样触摸4次。

在校准函数中,由于之前重复触摸了4下屏幕,触摸的4个点的AD值被存入到了pos_temp[4][2]数组中,然后算出(x1,y1),
(x2,y2)之间的距离d1和(x3,y3),(x4,y4)之间的距离d2,把这两个水平距离相除得到一个比值fac1;再计算出(x1,y1),(x3,y3)之间的距离d3和(x2,y2),(x4,y4)之间的距离d4,把这两竖直方向的距离相除,得到一个比值fac2.如果0.95<fac1<1.05且0.95<fac2<1.05则认为校准成功。否则显示触摸的AD值,不断校正。如果校准成功,就计算AD值与实际坐标之间的校准参数,并将这些校准参数存到EEPROM(断点可保存数据)中。每次重新上电系统初始化后再读出来。

xfac、yfac:每个AD点对应的像素点数目。(液晶理论宽度-40)/(x2-x1) 即液晶理论宽度点阵值/AD测量值

xoff、yoff:测量误差值。[液晶理论宽度点阵值 - 每AD值对应多少点阵*(AD测量值)]/2 = 测量误差值(理论值为 20 点阵,实际是有误差的)
————————————————