你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

STM32f103的电阻触摸屏的五点校正算法

[复制链接]
Savy1314 发布时间:2017-7-18 14:26
  STM32f103的电阻触摸屏的五点校正算法
由于电阻式触摸屏就是一种传感器,它利用压力感应进行控制,将矩形区域中触摸点(X,Y)的物理位置转换为代表 X坐标和 Y 坐标的电压。这里先引入两个概念,物理坐标和逻辑坐标。物理坐标指触摸屏上点的实际位置,通常以液晶上点的个数来度量。逻辑坐标指这点被触摸时A/D 转换后的坐标值。如图1,我们假定液晶最左下角为坐标轴原点A ,在液晶上任取一点B (十字线交叉中心),B 在X 方向距离A 10 个点,在Y 方向距离A20 个点,则这点的物理坐标为(10,20)。如果我们触摸这一点时得到的X 向A/D 转换值为100,Y 向A/D 转换值为200,则这点的逻辑坐标为(100,200)。

       常用的电阻式触摸屏矫正方法有两点校准法和三点校准法。本文这里介绍的是结合了不同的电阻式触摸屏矫正法的优化算法:五点校正法。其中主要的原理是使用4点矫正法的比例运算以及三点矫正法的基准点运算。五点校正法优势在于可以更加精确的计算出X和Y方向的比例缩放系数,同时提供了中心基准点,对于一些线性电阻系数比较差电阻式触摸屏有很好的校正功能。

       校正相关的变量主要有:

       x[5] , y[5] 五点定位的物理坐标

       xl[5] , yl[5] 五点定位的逻辑坐标

       KX , KY 横纵方向伸缩系数

       XLC , YLC 中心基点逻辑坐标

       XC , YC 中心基点物理坐标(数值采用LCD显示屏的物理长宽分辨率的一半)

       触摸屏常和点阵式液晶显示(LCD)屏叠加在一起配套使用,构成一个矩形的实际物理平面; 而由用户触摸的触摸点集合经过 A/D 转换器,得到具体显示坐标的集合,这个集合构成了一个逻辑平面。 由于存在误差,这两个平面并不重合,校准的作用就是要将逻辑平面映射到物理平面上,即得到触点在液晶屏上的位置坐标。 校准算法的中心思想也就是要建立这样一个映射函数现有的校准算法大多是基于线性校准, 即首先假定物理平面和逻辑平面之间的误差是线性误差,由旋转和偏移形成。
8.jpg

  x[5] , y[5] 五点定位的物理坐标是已知的,其中4点分别设置在LCD的角落,一点设置在LCD正中心,作为基准矫正点。校正关键点和距离布局如图。校正步骤如下:
        1. 通过先后点击LCD的4个角落的矫正点,获取4个角落的逻辑坐标值。
        2. 计算 s1’ = xl[2] - xl[1] 、 s3’ = xl[3] - xl[4] 、 s2’ = yl[3] - yl[2] 、 s4’ = yl[4] - yl[1]
            计算 s1 = x[2] - x[1] 、 s3 = x[3] - x[4] 、 s2 = y[3] - y[2] 、 s4 = y[4] - y[1],一般取点可以人为的设定s1 = s3 和 s2 = s4,以方便运算。
            计算 KX = ( s1’ + s3’ )/2/s1 、KY = ( s2’ + s4’ )/2/s2
        3. 点击LCD正中心,获取中心点的逻辑坐标,作为矫正的基准点。
        4. 完成以上步骤则校正完成。下次点击触摸屏的时候获取的逻辑值XL和YL,可根据公式转换成物理值:
            X = ( XL - XLC ) / KX + XC
            Y = ( YL - YLC ) / KY + YC
        换算出来的X , Y即是和LCD像素相对应的物理坐标值,方便对触屏响应程序做区域判别。

以下是校正程序:
  1. * 名    称:void LCD_Adjustd(void)
  2. * 功    能:校正电阻屏系数
  3. * 入口参数: null
  4. * 出口参数:无
  5. * 说    明:null
  6. * 调用方法:LCD_Adjustd();
  7. ****************************************************************************/  
  8. u8 LCD_Adjustd(void)  
  9. {  
  10.     EXTI_InitTypeDef EXTI_InitStructure;  
  11.       
  12.     EXTI_InitStructure.EXTI_Line    = EXTI_Line7;  
  13.     EXTI_InitStructure.EXTI_Mode    = EXTI_Mode_Interrupt;  //为中断请求  
  14.     EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling;//Falling下降沿 Rising上升  
  15.     EXTI_InitStructure.EXTI_LineCmd = DISABLE;  
  16.     EXTI_Init(&EXTI_InitStructure);  
  17.     //显示停止刷屏  
  18.     TIM_Cmd(TIM3, DISABLE);  //使能TIMx外设  
  19.       
  20.     LCD_Clear(White );  
  21.     LCD_printString(110,20, "Adjustd Begin" ,Black);  
  22.     delay_ms(5000);  
  23.     // 定第一个点  
  24.     LCD_Draw_Target(20, 20, Red);  
  25.     while( GPIO_ReadInputDataBit(GPIOG,GPIO_Pin_7));  
  26.     while( (1-GPIO_ReadInputDataBit(GPIOG,GPIO_Pin_7)))  
  27.     {  
  28.         x[0] = Read_XY(CMD_RDX);  
  29.         y[0] = Read_XY(CMD_RDY);   
  30.         LCD_ShowNum(150,80,x[0],Black);  
  31.         LCD_ShowNum(150,110,y[0],Black);  
  32.         delay_ms(200);  
  33.         LCD_Color_Fill(150,80,200,120, White);  
  34.     }  
  35.     // 定第二个点  
  36.     LCD_Draw_Target(300, 20, Red);  
  37.     LCD_Draw_Target(20, 20, White);  
  38.     while( GPIO_ReadInputDataBit(GPIOG,GPIO_Pin_7));  
  39.     while( (1-GPIO_ReadInputDataBit(GPIOG,GPIO_Pin_7)))  
  40.     {  
  41.         x[1] = Read_XY(CMD_RDX);  
  42.         y[1] = Read_XY(CMD_RDY);   
  43.         LCD_ShowNum(150,80,x[1],Black);  
  44.         LCD_ShowNum(150,110,y[1],Black);  
  45.         delay_ms(200);  
  46.         LCD_Color_Fill(150,80,200,120, White);  
  47.     }  
  48.     if(abs(y[1]-y[0]) >60)  
  49.     {  
  50.         LCD_Clear(White );  
  51.         LCD_printString(110,20, "Adjustd Fail" ,Black);  
  52.         delay_ms(5000);  
  53.         LCD_Clear(White );  
  54.         return 1;  
  55.     }  
  56.     // 定第三个点  
  57.     LCD_Draw_Target(20, 220, Red);  
  58.     LCD_Draw_Target(300, 20, White);  
  59.     while( GPIO_ReadInputDataBit(GPIOG,GPIO_Pin_7));  
  60.     while( (1-GPIO_ReadInputDataBit(GPIOG,GPIO_Pin_7)))  
  61.     {  
  62.         x[2] = Read_XY(CMD_RDX);  
  63.         y[2] = Read_XY(CMD_RDY);   
  64.         LCD_ShowNum(150,80,x[2],Black);  
  65.         LCD_ShowNum(150,110,y[2],Black);  
  66.         delay_ms(200);  
  67.         LCD_Color_Fill(150,80,200,120, White);  
  68.     }  
  69.     if(abs(x[2]-x[0]) >80)  
  70.     {  
  71.         LCD_Clear(White );  
  72.         LCD_printString(110,20, "Adjustd Fail" ,Black);  
  73.         delay_ms(5000);  
  74.         LCD_Clear(White );  
  75.         return 1;  
  76.     }  
  77.     // 定第四个点  
  78.     LCD_Draw_Target(300, 220, Red);  
  79.     LCD_Draw_Target(20, 220, White);  
  80.     while( GPIO_ReadInputDataBit(GPIOG,GPIO_Pin_7));  
  81.     while( (1-GPIO_ReadInputDataBit(GPIOG,GPIO_Pin_7)))  
  82.     {  
  83.         x[3] = Read_XY(CMD_RDX);  
  84.         y[3] = Read_XY(CMD_RDY);   
  85.         LCD_ShowNum(150,80,x[3],Black);  
  86.         LCD_ShowNum(150,110,y[3],Black);  
  87.         delay_ms(200);  
  88.         LCD_Color_Fill(150,80,200,120, White);  
  89.     }  
  90.     if((abs(y[2]-y[3]) >60) || (abs(x[1]-x[3]) >80))  
  91.     {  
  92.         LCD_Clear(White );  
  93.         LCD_printString(110,20, "Adjustd Fail" ,Black);  
  94.         delay_ms(5000);  
  95.         LCD_Clear(White );  
  96.         return 1;  
  97.     }  
  98.     // 定第五个点  
  99.     LCD_Draw_Target(160, 120, Red);  
  100.     LCD_Draw_Target(300, 220, White);  
  101.     while( GPIO_ReadInputDataBit(GPIOG,GPIO_Pin_7));  
  102.     while( (1-GPIO_ReadInputDataBit(GPIOG,GPIO_Pin_7)))  
  103.     {  
  104.         x[4] = Read_XY(CMD_RDX);  
  105.         y[4] = Read_XY(CMD_RDY);  
  106.         delay_ms(200);  
  107.     }  
  108.     //计算校正系数  
  109. //  KX  = ((abs(y[0]-y[2])/280+abs(y[1]-y[3])/280)/2);  
  110. //  KY  = ((abs(x[0]-x[1])/200+abs(x[2]-x[3])/200)/2);  
  111.     KX  = (((float)(y[0]-y[2])/280+(float)(y[1]-y[3])/280)/2);  
  112.     KY  = (((float)(x[0]-x[1])/200+(float)(x[2]-x[3])/200)/2);  
  113.     XC = 160;  
  114.     YC = 120;  
  115.     XLC  = y[4];  
  116.     YLC  = x[4];  
  117.       
  118.     // 定点完成  
  119.     LCD_Clear(White );  
  120.     LCD_printString(110,20, "Adjustd Done" ,Black);  
  121.     delay_ms(5000);  
  122.     LCD_Color_Fill(110,20,200,35, White);  
  123.     LCD_printString(110,20, "Testing" ,Black);  
  124.       
  125.     EXTI_InitStructure.EXTI_Line    = EXTI_Line7;  
  126.     EXTI_InitStructure.EXTI_Mode    = EXTI_Mode_Interrupt;  //为中断请求  
  127.     EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling;//Falling下降沿 Rising上升  
  128.     EXTI_InitStructure.EXTI_LineCmd = ENABLE;  
  129.     EXTI_Init(&EXTI_InitStructure);  
  130.     EXTI_ClearITPendingBit(EXTI_Line7);    //清除线路挂起位  
  131.       
  132.     //显示开始刷屏  
  133.     TIM_Cmd(TIM3, ENABLE);  //使能TIMx外设  
  134.       
  135.     Add_Button();  
  136.       
  137.     return 0;  
  138. }  
复制代码


评分

参与人数 1 ST金币 +2 收起 理由
MrJiu + 2 赞一个!

查看全部评分

收藏 4 评论5 发布时间:2017-7-18 14:26

举报

5个回答
huaiqiao 回答时间:2017-7-18 21:12:39
多谢分享,学习啦
MrJiu 回答时间:2017-7-19 08:41:42
正需要!!!
dsjsjf 回答时间:2017-7-19 13:05:28
学习了,我一般都是用的两点,第三点作为校验
epochal 回答时间:2017-7-28 19:20:45
谢谢分享!
五哥1 回答时间:2017-7-29 17:01:05
多谢分享,学习啦

所属标签

STM32团队

意法半导体微控制器和微处理器拥有广泛的产品线,包含低成本的8位单片机和基于ARM® Cortex®-M0、M0+、M3、M4、M33、M7及A7内核并具备丰富外设选择的32位微控制器及微处理器


最新内容

相似分享

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版