标题:
xpt2046.h和.c头文件 电阻触摸屏驱动下载
[打印本页]
作者:
wezern
时间:
2017-7-26 15:52
标题:
xpt2046.h和.c头文件 电阻触摸屏驱动下载
0.png
(42.34 KB, 下载次数: 133)
下载附件
2017-7-26 17:54 上传
附件两个代码为xpt2046.c和xpt2046.h,是常用的电阻触摸屏驱动
所有资料51hei提供下载:
xpt2046.zip
(4.65 KB, 下载次数: 197)
2017-7-26 17:54 上传
点击文件名下载附件
下载积分: 黑币 -5
单片机源程序如下:
#include "xpt2046.h"
#include "ili93xx.h"
#include "stdlib.h"
#include "math.h"
Pen_Holder Pen_Point;//定义笔实体
void delay_us(u32 us)
{
u32 time=100*us/7;
while(--time);
}
//SPI写数据
//向7846/7843/XPT2046/UH7843/UH7846写入1byte数据
void ADS_Write_Byte(u8 num)
{
u8 count=0;
for(count=0;count<8;count++)
{
if(num&0x80)
Set_TOUCH_TDIN
else
Clr_TOUCH_TDIN;
num<<=1;
Clr_TOUCH_TCLK;//上升沿有效
Set_TOUCH_TCLK;
}
}
//SPI读数据
//从7846/7843/XPT2046/UH7843/UH7846读取adc值
u16 ADS_Read_AD(u8 CMD)
{
u8 count=0;
u16 Num=0;
Clr_TOUCH_TCLK;//先拉低时钟
Clr_TOUCH_TCS; //选中ADS7843
ADS_Write_Byte(CMD);//发送命令字
delay_us(6);//ADS7846的转换时间最长为6us
Set_TOUCH_TCLK;//给1个时钟,清除BUSY
Clr_TOUCH_TCLK;
for(count=0;count<16;count++)
{
Num<<=1;
Clr_TOUCH_TCLK;//下降沿有效
Set_TOUCH_TCLK;
if(TOUCH_DOUT)
Num++;
}
Num>>=4; //只有高12位有效.
Set_TOUCH_TCS;//释放ADS7843
return(Num);
}
//读取一个坐标值
//连续读取READ_TIMES次数据,对这些数据升序排列,
//然后去掉最低和最高LOST_VAL个数,取平均值
#define READ_TIMES 15 //读取次数
#define LOST_VAL 5 //丢弃值
u16 ADS_Read_XY(u8 xy)
{
u16 i, j;
u16 buf[READ_TIMES];
u16 sum=0;
u16 temp;
for(i=0;i<READ_TIMES;i++)
{
buf[i]=ADS_Read_AD(xy);
}
for(i=0;i<READ_TIMES-1; i++)//排序
{
for(j=i+1;j<READ_TIMES;j++)
{
if(buf[i]>buf[j])//升序排列
{
temp=buf[i];
buf[i]=buf[j];
buf[j]=temp;
}
}
}
sum=0;
for(i=LOST_VAL;i<READ_TIMES-LOST_VAL;i++)sum+=buf[i];
temp=sum/(READ_TIMES-2*LOST_VAL);
return temp;
}
//带滤波的坐标读取
//最小值不能少于100.
u8 Read_ADS(u16 *x,u16 *y)
{
u16 xtemp,ytemp;
xtemp=ADS_Read_XY(CMD_RDX);
ytemp=ADS_Read_XY(CMD_RDY);
if(xtemp<100||ytemp<100)return 0;//读数失败
*x=xtemp;
*y=ytemp;
return 1;//读数成功
}
//2次读取ADS7846,连续读取2次有效的AD值,且这两次的偏差不能超过
//50,满足条件,则认为读数正确,否则读数错误.
//该函数能大大提高准确度
#define ERR_RANGE 50 //误差范围
u8 Read_ADS2(u16 *x,u16 *y)
{
u16 x1,y1;
u16 x2,y2;
u8 flag;
flag=Read_ADS(&x1,&y1);
if(flag==0)return(0);
flag=Read_ADS(&x2,&y2);
if(flag==0)return(0);
if(((x2<=x1&&x1<x2+ERR_RANGE)||(x1<=x2&&x2<x1+ERR_RANGE))//前后两次采样在+-50内
&&((y2<=y1&&y1<y2+ERR_RANGE)||(y1<=y2&&y2<y1+ERR_RANGE)))
{
*x=(x1+x2)/2;
*y=(y1+y2)/2;
return 1;
}else{
return 0;
}
}
//读取一次坐标值
//仅仅读取一次,知道PEN松开才返回!
u8 Read_TP_Once(void)
{
u8 t=0;
Pen_Int_Set(0);//关闭中断
Pen_Point.Key_Sta=Key_Up;
Read_ADS2(&Pen_Point.X,&Pen_Point.Y);
while(TOUCH_PEN==0&&t<=250)
{
t++;
delay_us(10000);
};
Pen_Int_Set(1);//开启中断
if(t>=250)
return 0;//按下2.5s 认为无效
else
return 1;
}
/**************************************与LCD部分有关的函数****************************************/
//画一个触摸点
//用来校准用的
void Drow_Touch_Point(u8 x,u16 y)
{
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的点
void Draw_Big_Point(u8 x,u16 y)
{
LCD_DrawPoint(x,y);//中心点
LCD_DrawPoint(x+1,y);
LCD_DrawPoint(x,y+1);
LCD_DrawPoint(x+1,y+1);
}
/***************************************************************************************************/
//转换结果
//根据触摸屏的校准参数来决定转换后的结果,保存在X0,Y0中
void Convert_Pos(void)
{
if(Read_ADS2(&Pen_Point.X,&Pen_Point.Y))
{
Pen_Point.X0=Pen_Point.xfac*Pen_Point.X+Pen_Point.xoff;
Pen_Point.Y0=Pen_Point.yfac*Pen_Point.Y+Pen_Point.yoff;
}
}
//PEN中断设置
//启用一个外部中断当触摸屏被按下后出发外部中断
//外部中断函数:
// 检测PEN脚的一个下降沿
// void EXTI1_IRQHandler(void)
// {
// Pen_Point.Key_Sta=Key_Down;//按键按下
// EXTI_ClearFlag(EXTI_Line1);//清除LINE1上的中断标志位
// }
void Pen_Int_Set(u8 en)
{
EXTI_InitTypeDef EXTI_InitStructure;
if(en)
{
EXTI_InitStructure.EXTI_Line = EXTI_Line1; //外部线路EXIT1
EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt; //设外外部中断模式:EXTI线路为中断请求
EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling; //外部中断触发沿选择:设置输入线路下降沿为中断请求
EXTI_InitStructure.EXTI_LineCmd = ENABLE; //使能外部中断新状态
EXTI_Init(&EXTI_InitStructure); //根据EXTI_InitStruct中指定的参数初始化外设EXTI寄存器
}else{
EXTI_InitStructure.EXTI_Line = EXTI_Line1; //外部线路EXIT1
EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt; //设外外部中断模式:EXTI线路为中断请求
EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling; //外部中断触发沿选择:设置输入线路下降沿为中断请求
EXTI_InitStructure.EXTI_LineCmd = DISABLE; //关闭外部中断新状态
EXTI_Init(&EXTI_InitStructure); //根据EXTI_InitStruct中指定的参数初始化外设EXTI寄存器
}
}
//触摸屏校准代码
//得到四个校准参数
void Touch_Adjust(void)
{
u16 pos_temp[4][2];//坐标缓存值
u8 cnt=0;
u16 d1,d2;
u32 tem1,tem2;
float fac;
cnt=0;
POINT_COLOR=BLUE;
BACK_COLOR =WHITE;
LCD_Clear(WHITE);//清屏
POINT_COLOR=RED;//红色
LCD_Clear(WHITE);//清屏
Drow_Touch_Point(20,20);//画点1
Pen_Point.Key_Sta=Key_Up;//消除触发信号
Pen_Point.xfac=0;//xfac用来标记是否校准过,所以校准之前必须清掉!以免错误
while(1)
{
if(Pen_Point.Key_Sta==Key_Down)//按键按下了
{
if(Read_TP_Once())//得到单次按键值
{
pos_temp[cnt][0]=Pen_Point.X;
pos_temp[cnt][1]=Pen_Point.Y;
cnt++;
}
switch(cnt)
{
case 1:
LCD_Clear(WHITE);//清屏
Drow_Touch_Point(220,20);//画点2
break;
case 2:
LCD_Clear(WHITE);//清屏
Drow_Touch_Point(20,300);//画点3
break;
case 3:
LCD_Clear(WHITE);//清屏
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)//不合格
……………………
…………限于本文篇幅 余下代码请从51黑下载附件…………
复制代码
作者:
wdliming
时间:
2018-8-14 10:51
谢谢分享~~~
作者:
lzm5523
时间:
2018-11-18 18:20
谢谢分享~~~~
作者:
stn13135
时间:
2019-3-6 09:49
谢谢分享~~~~
作者:
stn13135
时间:
2019-3-7 09:57
谢谢分享~~~~但要5黑币呀
作者:
GrandpaWang
时间:
2019-4-26 22:12
谢谢分享~~~~
作者:
huangchurong
时间:
2019-10-10 20:46
谢谢
作者:
FayeLiu
时间:
2019-11-13 14:43
谢谢分享~
作者:
wulingqing
时间:
2019-11-13 15:38
触摸屏驱动好资料,谢谢分享
作者:
qq_1634586006
时间:
2019-12-1 16:33
谢谢分享~~~
作者:
汪嘉凯
时间:
2020-10-25 14:23
谢谢楼主分享~先留言,黑币够了来下载一下
作者:
biren
时间:
2020-12-5 10:22
芯片比较早了,但基础知识多了解才能扎实,谢谢。
作者:
Pastorwei
时间:
2020-12-15 15:36
虽然芯片比较老了,但好多行业还是会用到我想多了解一下基础知道。谢谢楼主了
欢迎光临 (http://www.51hei.com/bbs/)
Powered by Discuz! X3.1