ref: c9389e2d62cc34902bc0f7b4aebd6d1f7c33f51c
dir: /sys/src/cmd/plot/libplot/machdep.c/
#include "mplot.h"
Image *offscreen;
static int buffer;
static Point
xlp(Point p)
{
	p.x += screen->r.min.x + 4 - offscreen->r.min.x;
	p.y += screen->r.min.y + 4 - offscreen->r.min.y;
	return p;
}
static Rectangle
xlr(Rectangle r)
{
	int dx, dy;
	dx = screen->r.min.x + 4 - offscreen->r.min.x;
	dy = screen->r.min.y + 4 - offscreen->r.min.y;
	r.min.x += dx;
	r.min.y += dy;
	r.max.x += dx;
	r.max.y += dy;
	return r;
}
/*
 * Clear the window from x0, y0 to x1, y1 (inclusive) to color c
 */
void
m_clrwin(int x0, int y0, int x1, int y1, int c)
{
	draw(offscreen, Rect(x0, y0, x1+1, y1+1), getcolor(c), nil, ZP);
	if(offscreen != screen && !buffer)
		draw(screen, xlr(Rect(x0, y0, x1+1, y1+1)), getcolor(c), nil, ZP);
}
/*
 * Draw circle at point p with radius rad in color c
 */
void
m_circ(Point p, int rad, int c)
{
	ellipse(offscreen, p, rad, rad, 0, getcolor(c), ZP);
	if (offscreen != screen && !buffer)
		ellipse(screen, p, rad, rad, 0, getcolor(c), ZP);
}
/*
 * Draw disc (filled circle) at point p with radius rad in color c
 */
void
m_disc(Point p, int rad, int c)
{
	fillellipse(offscreen, p, rad, rad, getcolor(c), ZP);
	if (offscreen != screen && !buffer)
		fillellipse(screen, p, rad, rad, getcolor(c), ZP);
}
/*
 * Draw text between pointers p and q with first character centered at x, y.
 * Use color c.  Centered if cen is non-zero, right-justified if right is non-zero.
 * Returns the y coordinate for any following line of text.
 */
int
m_text(int x, int y, char *p, char *q, int c, int cen, int right)
{
	Point tsize;
	tsize = stringsize(font, p);
	if(cen)
		x -= tsize.x/2;
	else if(right)
		x -= tsize.x;
	stringn(offscreen, Pt(x, y-tsize.y/2), getcolor(c), ZP, font, p, q-p);
	if(offscreen != screen && !buffer)
		stringn(screen, xlp(Pt(x, y-tsize.y/2)), getcolor(c), ZP, font, p, q-p);
	return y+tsize.y;
}
/*
 * draw point x, y
 */
void
m_dpt(double x, double y)
{
	Image *c;
	c = getcolor(e1->foregr);
	draw(offscreen, Rect(SCX(x), SCY(y), SCX(x)+1, SCY(y)+1), c, nil, ZP);
	if(offscreen != screen && !buffer)
		draw(screen, xlr(Rect(SCX(x), SCY(y), SCX(x)+1, SCY(y)+1)), c, nil, ZP);
}
/*
 * Draw the vector from x0, y0 to x1, y1 in color c.
 * Clipped by caller
 */
void
m_vector(int x0, int y0, int x1, int y1, int c)
{
	line(offscreen, Pt(x0, y0), Pt(x1, y1), Endsquare, Endsquare, 0, getcolor(c), ZP);
	if(offscreen != screen && !buffer)
		line(screen, xlp(Pt(x0, y0)), xlp(Pt(x1, y1)), Endsquare, Endsquare, 0, getcolor(c), ZP);
}
/*
 * Startup initialization
 */
void m_initialize(char*)
{
	static int once;
	int dx, dy;
	if(once)
		return;
	once = 1;
	if(initdraw(nil, nil, "plot") < 0)
		sysfatal("initdraw: %r");
/////	einit(Emouse);
	offscreen = allocimage(display, insetrect(screen->r, 4), screen->chan, 0, -1);
	if(offscreen == nil)
		sysfatal("Can't double buffer\n");
	clipminx = mapminx = screen->r.min.x+4;
	clipminy = mapminy = screen->r.min.y+4;
	clipmaxx = mapmaxx = screen->r.max.x-5;
	clipmaxy = mapmaxy = screen->r.max.y-5;
	dx = clipmaxx-clipminx;
	dy = clipmaxy-clipminy;
	if(dx>dy){
		mapminx += (dx-dy)/2;
		mapmaxx = mapminx+dy;
	}
	else{
		mapminy += (dy-dx)/2;
		mapmaxy = mapminy+dx;
	}
}
/*
 * Clean up when finished
 */
void m_finish(void)
{
	m_swapbuf();
}
void m_swapbuf(void)
{
	draw(screen, insetrect(screen->r, 4), offscreen, nil, offscreen->r.min);
	flushimage(display, 1);
}
void m_dblbuf(void)
{
	buffer = 1;
}
/*
 * Use cache to avoid repeated allocation.
 */
struct{
	int		v;
	Image	*i;
}icache[32];
Image*
getcolor(int v)
{
	Image *i;
	int j;
	for(j=0; j<nelem(icache); j++)
		if(icache[j].v==v && icache[j].i!=nil)
			return icache[j].i;
	i = allocimage(display, Rect(0, 0, 1, 1), RGB24, 1, v);
	if(i == nil)
		sysfatal("plot: can't allocate image for color: %r");
	for(j=0; j<nelem(icache); j++)
		if(icache[j].i == nil){
			icache[j].v = v;
			icache[j].i = i;
			break;
		}
if(j == nelem(icache))sysfatal("icache: too small");
	return i;
}