code: plan9front

ref: 5622b0bbd878dbc34045cc6fd37cffa64461eabe
dir: /sys/src/cmd/bitsy/pencal.c/

View raw version
#include <u.h>
#include <libc.h>
#include <draw.h>

int debug = 0;

struct Cal {
	double scalex;
	double transx;
	double scaley;
	double transy;
} cal;

/* Calibrate the pen */

Point pts[5];

Point pens[5];

Point up	= {  0, -10};
Point down	= {  0,  10};
Point left	= {-10,   0};
Point right	= { 10,   0};

Point
pen2scr(Point p)
{
	return Pt(p.x * cal.scalex / 65536.0 + cal.transx,
			  p.y * cal.scaley / 65536.0 + cal.transy);
}

Point
scr2pen(Point p)
{
	return Pt((p.x - cal.transx) * 65536.0 / cal.scalex,
			  (p.y + cal.transy) * 65536.0 / cal.scaley);
}

void
cross(Point p) {
	draw(screen, screen->r, display->white, nil, ZP);
	line(screen, addpt(p, up), addpt(p, down), Endsquare, Endsquare, 0, display->black, ZP);
	line(screen, addpt(p, left), addpt(p, right), Endsquare, Endsquare, 0, display->black, ZP);
	flushimage(display, 1);
}

Point
getmouse(int mouse) {
	static char buf[50];
	int x, y, b, n;
	Point p;

	n = 0;
	p.x = p.y = 0;
	do{
		if (read(mouse, buf, 48) != 48)
			sysfatal("read: %r");
		if (debug > 1) fprint(2, "%s\n", buf);
		if (buf[0] == 'r') {
			b = 1;
			continue;
		}
		if (buf[0] != 'm')
			sysfatal("mouse message: %s", buf);
		x = strtol(buf+1, nil, 10);
		y = strtol(buf+13, nil, 10);
		b = strtol(buf+25, nil, 10);
		p.x = (n * p.x + x)/(n+1);
		p.y = (n * p.y + y)/(n+1);
		n++;
	}while (b & 0x7);
	return p;
}

void
main(int argc, char **argv) {
	int i, mouse, mousectl, wctl, ntries;
	Rectangle r, oldsize;

	ARGBEGIN{
	case 'd':
		debug = 1;
		break;
	}ARGEND;

	if((mouse = open("/dev/mouse", OREAD)) < 0
	 && (mouse = open("#m/mouse", OREAD)) < 0)
		sysfatal("#m/mouse: %r");

	mousectl = open("#m/mousectl", ORDWR);
	if(mousectl < 0)
		sysfatal("#m/mousectl: %r");

	if(initdraw(nil, nil, "calibrate") < 0)
		sysfatal("initdraw: %r");

	wctl = -1;
	for(ntries = 0; ntries < 3; ntries++){
		r = insetrect(display->image->r, 2);

		if(wctl < 0)
			wctl = open("/dev/wctl", ORDWR);
		if(wctl >= 0){
			char buf[4*12+1];

			buf[48] = 0;
			oldsize = screen->r;
			if(fprint(wctl, "resize -r %R", r) <= 0)
				sysfatal("write /dev/wctl, %r");
			if(getwindow(display, Refbackup) < 0)
				sysfatal("getwindow");
			if(debug) fprint(2, "resize: %R\n", screen->r);

			if(read(mousectl, buf, 48) != 48)
				sysfatal("read mousectl: %r");
			if(debug > 1) fprint(2, "mousectl %s\n", buf);
			if(buf[0] != 'c')
				sysfatal("mousectl message: %s", buf);
			cal.scalex = strtol(buf+ 1, nil, 10);
			cal.scaley = strtol(buf+13, nil, 10);
			cal.transx = strtol(buf+25, nil, 10);
			cal.transy = strtol(buf+37, nil, 10);
		}else{
			fprint(mousectl, "calibrate");
			cal.scalex = 1<<16;
			cal.scaley = 1<<16;
			cal.transx = 0;
			cal.transy = 0;
		}

		fprint(2, "calibrate %ld %ld %ld %ld (old)\n",
			(long)cal.scalex, (long)cal.scaley, (long)cal.transx, (long)cal.transy);

		if(debug)
			fprint(2, "screen coords %R\n", screen->r);
		pts[0] = addpt(Pt( 16,  16), screen->r.min);
		pts[1] = addpt(Pt( 16, -16), Pt(screen->r.min.x, screen->r.max.y));
		pts[2] = addpt(Pt(-16,  16), Pt(screen->r.max.x, screen->r.min.y));
		pts[3] = addpt(Pt(-16, -16), screen->r.max);
		pts[4] = Pt((screen->r.max.x + screen->r.min.x)/2,
					(screen->r.max.y + screen->r.min.y)/2);;

		for(i = 0; i < 4; i++){
			cross(pts[i]);
			pens[i] = scr2pen(getmouse(mouse));
			if (debug) fprint(2, "%P\n", pens[i]);
		}

		cal.scalex = (pts[2].x + pts[3].x - pts[0].x - pts[1].x) * 65536.0 /
				  (pens[2].x + pens[3].x - pens[0].x - pens[1].x);
		cal.scaley = (pts[1].y + pts[3].y - pts[0].y - pts[2].y) * 65536.0 /
				  (pens[1].y + pens[3].y - pens[0].y - pens[2].y);
		cal.transx = pts[0].x - (pens[0].x*cal.scalex) / 65536.0;
		cal.transy = pts[0].y - (pens[0].y*cal.scaley) / 65536.0;

		if(debug)
			fprint(2, "scale [%f, %f], trans [%f, %f]\n",
				cal.scalex, cal.scaley, cal.transx, cal.transy);

		fprint(mousectl, "calibrate %ld %ld %ld %ld",
			(long)cal.scalex, (long)cal.scaley, (long)cal.transx, (long)cal.transy);

		cross(pts[4]);
		pens[4] = getmouse(mouse);

		draw(screen, screen->r, display->white, nil, ZP);
		flushimage(display, 1);

		if(abs(pts[4].x-pens[4].x) <= 4 && abs(pts[4].y-pens[4].y) <= 4){
			fprint(2, "Calibration ok: %P -> %P\n", pts[4], pens[4]);
			break;
		}
		fprint(2, "Calibration inaccurate: %P -> %P\n", pts[4], pens[4]);
	}
	print("calibrate='%ld %ld %ld %ld'\n",
		(long)cal.scalex, (long)cal.scaley, (long)cal.transx, (long)cal.transy);

	if(debug)
		for(i = 0; i < 4; i++){
			cross(pts[i]);
			pens[i] = getmouse(mouse);
			if (debug) fprint(2, "%P\n", pens[i]);
		}

	if(wctl >= 0)
		fprint(wctl, "resize -r %R", oldsize);

}