touch驱动代码.docx
- 文档编号:30230204
- 上传时间:2023-08-07
- 格式:DOCX
- 页数:20
- 大小:20.34KB
touch驱动代码.docx
《touch驱动代码.docx》由会员分享,可在线阅读,更多相关《touch驱动代码.docx(20页珍藏版)》请在冰豆网上搜索。
touch驱动代码
#include"touch.h"
#include"lcd.h"
#include"delay.h"
#include"stdlib.h"
#include"math.h"
#include"24cxx.h"
//////////////////////////////////////////////////////////////////////////////////
reserved
//********************************************************************************
//V2.0修改说明
//增加对电容触摸屏的支持(需要添加:
ctiic.c和ott2001a.c两个文件)
//////////////////////////////////////////////////////////////////////////////////
_m_tp_devtp_dev=
{
TP_Init,
TP_Scan,
TP_Adjust,
0,
0,
0,
0,
0,
0,
0,
0,
};
//默认为touchtype=0的数据.
u8CMD_RDX=0XD0;
u8CMD_RDY=0X90;
//SPI写数据
//向触摸屏IC写入1byte数据
//num:
要写入的数据
voidTP_Write_Byte(u8num)
{
u8count=0;
for(count=0;count<8;count++)
{
if(num&0x80)TDIN=1;
elseTDIN=0;
num<<=1;
TCLK=0;
TCLK=1;//上升沿有效
}
}
//SPI读数据
//从触摸屏IC读取adc值
//CMD:
指令
//返回值:
读到的数据
u16TP_Read_AD(u8CMD)
{
u8count=0;
u16Num=0;
TCLK=0;//先拉低时钟
TDIN=0;//拉低数据线
TCS=0;//选中触摸屏IC
TP_Write_Byte(CMD);//发送命令字
delay_us(6);//ADS7846的转换时间最长为6us
TCLK=0;
delay_us
(1);
TCLK=1;//给1个时钟,清除BUSY
TCLK=0;
for(count=0;count<16;count++)//读出16位数据,只有高12位有效
{
Num<<=1;
TCLK=0;//下降沿有效
TCLK=1;
if(DOUT)Num++;
}
Num>>=4;//只有高12位有效.
TCS=1;//释放片选
return(Num);
}
//读取一个坐标值(x或者y)
//连续读取READ_TIMES次数据,对这些数据升序排列,
//然后去掉最低和最高LOST_VAL个数,取平均值
//xy:
指令(CMD_RDX/CMD_RDY)
//返回值:
读到的数据
#defineREAD_TIMES5//读取次数
#defineLOST_VAL1//丢弃值
u16TP_Read_XOY(u8xy)
{
u16i,j;
u16buf[READ_TIMES];
u16sum=0;
u16temp;
for(i=0;i for(i=0;i { for(j=i+1;j { if(buf[i]>buf[j])//升序排列 { temp=buf[i]; buf[i]=buf[j]; buf[j]=temp; } } } sum=0; for(i=LOST_VAL;i temp=sum/(READ_TIMES-2*LOST_VAL); returntemp; } //读取x,y坐标 //最小值不能少于100. //x,y: 读取到的坐标值 //返回值: 0,失败;1,成功。 u8TP_Read_XY(u16*x,u16*y) { u16xtemp,ytemp; xtemp=TP_Read_XOY(CMD_RDX); ytemp=TP_Read_XOY(CMD_RDY); //if(xtemp<100||ytemp<100)return0;//读数失败 *x=xtemp; *y=ytemp; return1;//读数成功 } //连续2次读取触摸屏IC,且这两次的偏差不能超过 //ERR_RANGE,满足条件,则认为读数正确,否则读数错误. //该函数能大大提高准确度 //x,y: 读取到的坐标值 //返回值: 0,失败;1,成功。 #defineERR_RANGE50//误差范围 u8TP_Read_XY2(u16*x,u16*y) { u16x1,y1; u16x2,y2; u8flag; flag=TP_Read_XY(&x1,&y1); if(flag==0)return(0); flag=TP_Read_XY(&x2,&y2); if(flag==0)return(0); if(((x2<=x1&&x1 &&((y2<=y1&&y1 { *x=(x1+x2)/2; *y=(y1+y2)/2; return1; }elsereturn0; } ////////////////////////////////////////////////////////////////////////////////// //与LCD部分有关的函数 //画一个触摸点 //用来校准用的 //x,y: 坐标 //color: 颜色 voidTP_Drow_Touch_Point(u16x,u16y,u16color) { POINT_COLOR=color; LCD_DrawLine(x-12,y,x+13,y);//横线 LCD_DrawLine(x,y-12,x,y+13);//竖线 LCD_DrawPoint(x+1,y+1); LCD_DrawPoint(x-1,y+1); LCD_DrawPoint(x+1,y-1); LCD_DrawPoint(x-1,y-1); Draw_Circle(x,y,6);//画中心圈 } //画一个大点(2*2的点) //x,y: 坐标 //color: 颜色 voidTP_Draw_Big_Point(u16x,u16y,u16color) { POINT_COLOR=color; LCD_DrawPoint(x,y);//中心点 LCD_DrawPoint(x+1,y); LCD_DrawPoint(x,y+1); LCD_DrawPoint(x+1,y+1); } ////////////////////////////////////////////////////////////////////////////////// //触摸按键扫描 //tp: 0,屏幕坐标;1,物理坐标(校准等特殊场合用) //返回值: 当前触屏状态. //0,触屏无触摸;1,触屏有触摸 u8TP_Scan(u8tp) { if(PEN==0)//有按键按下 { if(tp)TP_Read_XY2(&tp_dev.x[0],&tp_dev.y[0]);//读取物理坐标 elseif(TP_Read_XY2(&tp_dev.x[0],&tp_dev.y[0]))//读取屏幕坐标 { tp_dev.x[0]=tp_dev.xfac*tp_dev.x[0]+tp_dev.xoff;//将结果转换为屏幕坐标 tp_dev.y[0]=tp_dev.yfac*tp_dev.y[0]+tp_dev.yoff; } if((tp_dev.sta&TP_PRES_DOWN)==0)//之前没有被按下 { tp_dev.sta=TP_PRES_DOWN|TP_CATH_PRES;//按键按下 tp_dev.x[4]=tp_dev.x[0];//记录第一次按下时的坐标 tp_dev.y[4]=tp_dev.y[0]; } }else { if(tp_dev.sta&TP_PRES_DOWN)//之前是被按下的 { tp_dev.sta&=~(1<<7);//标记按键松开 }else//之前就没有被按下 { tp_dev.x[4]=0; tp_dev.y[4]=0; tp_dev.x[0]=0xffff; tp_dev.y[0]=0xffff; } } returntp_dev.sta&TP_PRES_DOWN;//返回当前的触屏状态 } ////////////////////////////////////////////////////////////////////////// //保存在EEPROM里面的地址区间基址,占用13个字节(RANGE: SAVE_ADDR_BASE~SAVE_ADDR_BASE+12) #defineSAVE_ADDR_BASE40 //保存校准参数 voidTP_Save_Adjdata(void) { s32temp; //保存校正结果! temp=tp_dev.xfac*100000000;//保存x校正因素 AT24CXX_WriteLenByte(SAVE_ADDR_BASE,temp,4); temp=tp_dev.yfac*100000000;//保存y校正因素 AT24CXX_WriteLenByte(SAVE_ADDR_BASE+4,temp,4); //保存x偏移量 AT24CXX_WriteLenByte(SAVE_ADDR_BASE+8,tp_dev.xoff,2); //保存y偏移量 AT24CXX_WriteLenByte(SAVE_ADDR_BASE+10,tp_dev.yoff,2); //保存触屏类型 AT24CXX_WriteOneByte(SAVE_ADDR_BASE+12,tp_dev.touchtype); temp=0X0A;//标记校准过了 AT24CXX_WriteOneByte(SAVE_ADDR_BASE+13,temp); } //得到保存在EEPROM里面的校准值 //返回值: 1,成功获取数据 //0,获取失败,要重新校准 u8TP_Get_Adjdata(void) { s32tempfac; tempfac=AT24CXX_ReadOneByte(SAVE_ADDR_BASE+13);//读取标记字,看是否校准过! if(tempfac==0X0A)//触摸屏已经校准过了 { tempfac=AT24CXX_ReadLenByte(SAVE_ADDR_BASE,4); tp_dev.xfac=(float)tempfac/100000000;//得到x校准参数 tempfac=AT24CXX_ReadLenByte(SAVE_ADDR_BASE+4,4); tp_dev.yfac=(float)tempfac/100000000;//得到y校准参数 //得到x偏移量 tp_dev.xoff=AT24CXX_ReadLenByte(SAVE_ADDR_BASE+8,2); //得到y偏移量 tp_dev.yoff=AT24CXX_ReadLenByte(SAVE_ADDR_BASE+10,2); tp_dev.touchtype=AT24CXX_ReadOneByte(SAVE_ADDR_BASE+12);//读取触屏类型标记 if(tp_dev.touchtype)//X,Y方向与屏幕相反 { CMD_RDX=0X90; CMD_RDY=0XD0; }else//X,Y方向与屏幕相同 { CMD_RDX=0XD0; CMD_RDY=0X90; } return1; } return0; } //提示字符串 constu8*TP_REMIND_MSG_TBL="Pleaseusethestylusclickthecrossonthescreen.Thecrosswillalwaysmoveuntilthescreenadjustmentiscompleted."; //提示校准结果(各个参数) voidTP_Adj_Info_Show(u16x0,u16y0,u16x1,u16y1,u16x2,u16y2,u16x3,u16y3,u16fac) { POINT_COLOR=RED; LCD_ShowString(40,160,lcddev.width,lcddev.height,16,"x1: "); LCD_ShowString(40+80,160,lcddev.width,lcddev.height,16,"y1: "); LCD_ShowString(40,180,lcddev.width,lcddev.height,16,"x2: "); LCD_ShowString(40+80,180,lcddev.width,lcddev.height,16,"y2: "); LCD_ShowString(40,200,lcddev.width,lcddev.height,16,"x3: "); LCD_ShowString(40+80,200,lcddev.width,lcddev.height,16,"y3: "); LCD_ShowString(40,220,lcddev.width,lcddev.height,16,"x4: "); LCD_ShowString(40+80,220,lcddev.width,lcddev.height,16,"y4: "); LCD_ShowString(40,240,lcddev.width,lcddev.height,16,"facis: "); LCD_ShowNum(40+24,160,x0,4,16);//显示数值 LCD_ShowNum(40+24+80,160,y0,4,16);//显示数值 LCD_ShowNum(40+24,180,x1,4,16);//显示数值 LCD_ShowNum(40+24+80,180,y1,4,16);//显示数值 LCD_ShowNum(40+24,200,x2,4,16);//显示数值 LCD_ShowNum(40+24+80,200,y2,4,16);//显示数值 LCD_ShowNum(40+24,220,x3,4,16);//显示数值 LCD_ShowNum(40+24+80,220,y3,4,16);//显示数值 LCD_ShowNum(40+56,lcddev.width,fac,3,16);//显示数值,该数值必须在95~105范围之内. } //触摸屏校准代码 //得到四个校准参数 voidTP_Adjust(void) { u16pos_temp[4][2];//坐标缓存值 u8cnt=0; u16d1,d2; u32tem1,tem2; floatfac; u16outtime=0; cnt=0; POINT_COLOR=BLUE; BACK_COLOR=WHITE; LCD_Clear(WHITE);//清屏 POINT_COLOR=RED;//红色 LCD_Clear(WHITE);//清屏 POINT_COLOR=BLACK; LCD_ShowString(40,40,160,100,16,(u8*)TP_REMIND_MSG_TBL);//显示提示信息 TP_Drow_Touch_Point(20,20,RED);//画点1 tp_dev.sta=0;//消除触发信号 tp_dev.xfac=0;//xfac用来标记是否校准过,所以校准之前必须清掉! 以免错误 while (1)//如果连续10秒钟没有按下,则自动退出 { tp_dev.scan (1);//扫描物理坐标 if((tp_dev.sta&0xc0)==TP_CATH_PRES)//按键按下了一次(此时按键松开了.) { outtime=0; tp_dev.sta&=~(1<<6);//标记按键已经被处理过了. pos_temp[cnt][0]=tp_dev.x[0]; pos_temp[cnt][1]=tp_dev.y[0]; cnt++; switch(cnt) { case1: TP_Drow_Touch_Point(20,20,WHITE);//清除点1 TP_Drow_Touch_Point(lcddev.width-20,20,RED);//画点2 break; case2: TP_Drow_Touch_Point(lcddev.width-20,20,WHITE);//清除点2 TP_Drow_Touch_Point(20,lcddev.height-20,RED);//画点3 break; case3: TP_Drow_Touch_Point(20,lcddev.height-20,WHITE);//清除点3 TP_Drow_Touch_Point(lcddev.width-20,lcddev.height-20,RED);//画点4 break; case4: //全部四个点已经得到 //对边相等 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; TP_Drow_Touch_Point(lcddev.width-20,lcddev.height-20,WHITE);//清除点4 TP_Drow_Touch_Point(20,20,RED);//画点1 TP_Adj_Info_Show(pos_temp[0][0],pos_temp[0][1],pos_temp[1][0],pos_temp[1][1],pos_temp[2][0],pos_temp[2][1],pos_temp[3][0],pos_temp[3][1],fac*100);//显示数据 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; TP_Drow_Touch_Point(lcddev.width-20,lcddev.height-20,WHITE);//清除点4 TP_Drow_Touch_Point(20,20,RED);//画点1 TP_Adj_Info_Show(pos_temp[0][0],pos_temp[0][1],pos_temp[1][0],pos_temp[1][1],pos_temp[2][0],pos_temp[2][1],pos_temp[3][0],pos_temp[3][1],fac*100);//显示数据 con
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- touch 驱动 代码