ref: 0231537cbac846087f5ac132aba5c97ec9d4cffa
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; }