今天做了android下触摸屏的校准,注意触摸屏可能有不一样,我们目前是用的是单点的电阻是触摸屏,android 平台是 2.1
下面来说下触摸屏的校准
client 采集数据:计算校对系数,写入文件 /data/etc/xxxx 通过propty来设置状态 InputDevice.java 根据propty 来读取校对的系数来校对数据。
校对数据模型采用 M=ax+by+k1; N=cx+dy+k2,其中 M,N为经校对系数计算后的准确的触摸屏位置。
那么我们要做的是通过数据采集来计算a,b,k1,c,d,k2,对于每一维的坐标,都是三个系数,因此我们采集三个有效的数据点。成而得到
M1=ax1+by1+k1 ; N1=cx1+dy1+k2
M2=ax2+by2+k1 ; N2=cx2+dy2+k2
M3=ax2+by2+k1 ; N3=cx2+dy2+k2,其中(M1,N1),(M2,N2),(M3,N3),是你校对的标准数据,我的代码里取
做上角(32,32),右上脚(SCREEN_WIDTH-32,32),右下脚(32,SCREEN_HEIGHT-32),
(x1,y1),(x2,y2),(x3,y3)分别是获取的没校准前的 触摸屏的数据。
那么可以解得a,b,k1,c,d,k2,让后将这6个数据写入到/data/et/xxx文件中。
然后InputDevice.java读取propty的状态来决定是否使用校对系数.。
当触摸屏设置完后就会设置为done
当InputDevie 发现propty为done的时候读取校对系数建立校对模型。
有触摸屏传来的数据(x,y),那么由公式,X准=ax+by+k1; Y准=cx+dy+k2; 试验验证准有效
代码分析:
界面代码 TouchCalibration.java,它很简单,一个全屏的activty
模型M的取值范围
int xList[] = {
32, UI_SCREEN_WIDTH - 32, 32, UI_SCREEN_WIDTH - 32, UI_SCREEN_WIDTH / 2
};
模型N的取值范围
int yList[] = {
32, 32, UI_SCREEN_HEIGHT - 32, UI_SCREEN_HEIGHT - 32, UI_SCREEN_HEIGHT / 2
};
为五个参考点的坐标,其实只用了3个点,哈哈(还没搞明白五个点的数学模型,如果有知道的,希望能不吝赐教)
重载 onTouchEvent的接口,然后活去取三个点的实际触摸到的坐标
x1,y1,x2,y2,x3,y3,而这个准确的位置分别对应了xList[0],yList[0],xList[1],yList[1],xList[2],yList[2],
取道数据后,用Calibrate.java来计算出 a,b,k1,c,d,k2,三个未知数三个一次方程求解,
主要函数在perform_calibration 中
a=((y3-y2)*(M2-M1)-(M3-M2)*(y2-y1))/((y3-y2)*(x2-x1)-(y2-y1)*(x3-x2))
其它的可以自己计算出来,
float M1,M2,M3,N1,N2,N3;
float a,b,k1,c,d,k2;
float x1,y1,x2,y2,x3,y3;
float mul,div;
float scaling = (float)65536.0;
M1=cal.xfb[0];
N1=cal.yfb[0];
M2=cal.xfb[1];
N2=cal.yfb[1];
M3=cal.xfb[2];
N3=cal.yfb[2];
x1=cal.x[0];
y1=cal.y[0];
x2=cal.x[1];
y2=cal.y[1];
x3=cal.x[2];
y3=cal.y[2];
div=((y3-y2)*(x2-x1)-(y2-y1)*(x3-x2));
mul=((y3-y2)*(M2-M1)-(M3-M2)*(y2-y1));
a=mul/div;
div=((x3-x2)*(y2-y1)-(x2-x1)*(y3-y2));
mul=(x3-x2)*(M2-M1)-(x2-x1)*(M3-M2);
b=mul/div;
k1=M1-a*x1-b*y1;
div=((y3-y2)*(x2-x1)-(y2-y1)*(x3-x2));
mul=((y3-y2)*(N2-N1)-(N3-N2)*(y2-y1));
c=mul/div;
div=((x3-x2)*(y2-y1)-(x2-x1)*(y3-y2));
mul=(x3-x2)*(N2-N1)-(x2-x1)*(N3-N2);
d=mul/div;
k2=N1-c*x1-d*y1;
Log.d("[hd debug]",a+":"+b+":"+k1+":"+c+":"+d+":"+k2+":");
cal.a[0]=(int)(a*scaling);
cal.a[1]=(int)(b*scaling);
cal.a[2]=(int)(k1*scaling);
cal.a[3]=(int)(c*scaling);
cal.a[4]=(int)(d*scaling);
cal.a[5]=(int)(k2*scaling);
cal.a[6]=(int)(scaling);
return true;
然后将三计算出来的值写入文件/data/etc/pointercal中,并设置ts.config.calibrate propty为done,
ts.config.calibrate propty是InputDevice.java,与触摸校对程序的通讯标志。
InputDevice.java 中修改你的generateAbsMotion函数,据说android 2.1支持多点触摸,我们用电阻式,所以没有去追究。
(我没有看懂)InputDevice.java代码,修改generateAbsMotion只能满足我的项目的需求,请仔细思考这里。
String prop = SystemProperties.get("ts.config.calibrate", "noset");
if (prop.equalsIgnoreCase("start")){
Log.d("XXW prop", prop);
Log.d("XXW", "prop.equalsIgnoreCase start");
device.tInfo = null;
}else if (prop.equalsIgnoreCase("done")){
Log.d("XXW prop", prop);
Log.d("XXW", "prop.equalsIgnoreCase done");
readCalibrate();
device.tInfo=tInfo;
SystemProperties.set("ts.config.calibrate", "end");
}else{
Log.d("XXW prop", prop);
Log.d("XXW", "prop.equalsIgnoreCase else");
}
这里根据propty设置是否采用校验系数,
如果采用的话,那么就使用隐射模型:
if (absX != null) {
if (device.tInfo != null){
reportData[j + MotionEvent.SAMPLE_X] = (device.tInfo.x1 * x + device.tInfo.y1 * y + device.tInfo.z1);
}
}
if (absY != null) {
if (device.tInfo != null){
reportData[j + MotionEvent.SAMPLE_Y] = (device.tInfo.x2 * x + device.tInfo.y2 * y + device.tInfo.z2);
}
}
来计算出新的坐标,
代码将在我的资源里给出,有心趣的同学自己实现,没有兴趣的同学花点分,去下载资源源码也是可以的。