xref: /plan9-contrib/sys/src/libframe/frdraw.c (revision 7dd7cddf99dd7472612f1413b4da293630e6b1bc)
13e12c5d1SDavid du Colombier #include <u.h>
23e12c5d1SDavid du Colombier #include <libc.h>
3*7dd7cddfSDavid du Colombier #include <draw.h>
4*7dd7cddfSDavid du Colombier #include <thread.h>
5*7dd7cddfSDavid du Colombier #include <mouse.h>
63e12c5d1SDavid du Colombier #include <frame.h>
73e12c5d1SDavid du Colombier 
83e12c5d1SDavid du Colombier void
93e12c5d1SDavid du Colombier _frredraw(Frame *f, Point pt)
103e12c5d1SDavid du Colombier {
113e12c5d1SDavid du Colombier 	Frbox *b;
123e12c5d1SDavid du Colombier 	int nb;
13*7dd7cddfSDavid du Colombier 	static int x;
14*7dd7cddfSDavid du Colombier 
153e12c5d1SDavid du Colombier 	for(nb=0,b=f->box; nb<f->nbox; nb++, b++){
163e12c5d1SDavid du Colombier 		_frcklinewrap(f, &pt, b);
17*7dd7cddfSDavid du Colombier 		if(b->nrune >= 0){
18*7dd7cddfSDavid du Colombier 			string(f->b, pt, f->cols[TEXT], ZP, f->font, (char *)b->ptr);
19*7dd7cddfSDavid du Colombier 		}
203e12c5d1SDavid du Colombier 		pt.x += b->wid;
213e12c5d1SDavid du Colombier 	}
223e12c5d1SDavid du Colombier }
233e12c5d1SDavid du Colombier 
24*7dd7cddfSDavid du Colombier static int
25*7dd7cddfSDavid du Colombier nbytes(char *s0, int nr)
26*7dd7cddfSDavid du Colombier {
27*7dd7cddfSDavid du Colombier 	char *s;
28*7dd7cddfSDavid du Colombier 	Rune r;
29*7dd7cddfSDavid du Colombier 
30*7dd7cddfSDavid du Colombier 	s = s0;
31*7dd7cddfSDavid du Colombier 	while(--nr >= 0)
32*7dd7cddfSDavid du Colombier 		s += chartorune(&r, s);
33*7dd7cddfSDavid du Colombier 	return s-s0;
34*7dd7cddfSDavid du Colombier }
35*7dd7cddfSDavid du Colombier 
36*7dd7cddfSDavid du Colombier void
37*7dd7cddfSDavid du Colombier frdrawsel(Frame *f, Point pt, ulong p0, ulong p1, int issel)
38*7dd7cddfSDavid du Colombier {
39*7dd7cddfSDavid du Colombier 	Image *back, *text;
40*7dd7cddfSDavid du Colombier 
41*7dd7cddfSDavid du Colombier 	if(f->ticked)
42*7dd7cddfSDavid du Colombier 		frtick(f, frptofchar(f, f->p0), 0);
43*7dd7cddfSDavid du Colombier 
44*7dd7cddfSDavid du Colombier 	if(p0 == p1){
45*7dd7cddfSDavid du Colombier 		frtick(f, pt, issel);
46*7dd7cddfSDavid du Colombier 		return;
47*7dd7cddfSDavid du Colombier 	}
48*7dd7cddfSDavid du Colombier 
49*7dd7cddfSDavid du Colombier 	if(issel){
50*7dd7cddfSDavid du Colombier 		back = f->cols[HIGH];
51*7dd7cddfSDavid du Colombier 		text = f->cols[HTEXT];
52*7dd7cddfSDavid du Colombier 	}else{
53*7dd7cddfSDavid du Colombier 		back = f->cols[BACK];
54*7dd7cddfSDavid du Colombier 		text = f->cols[TEXT];
55*7dd7cddfSDavid du Colombier 	}
56*7dd7cddfSDavid du Colombier 
57*7dd7cddfSDavid du Colombier 	frdrawsel0(f, pt, p0, p1, back, text);
58*7dd7cddfSDavid du Colombier }
59*7dd7cddfSDavid du Colombier 
60*7dd7cddfSDavid du Colombier void
61*7dd7cddfSDavid du Colombier frdrawsel0(Frame *f, Point pt, ulong p0, ulong p1, Image *back, Image *text)
62*7dd7cddfSDavid du Colombier {
63*7dd7cddfSDavid du Colombier 	Frbox *b;
64*7dd7cddfSDavid du Colombier 	int nb, nr, w, x, trim;
65*7dd7cddfSDavid du Colombier 	Point qt;
66*7dd7cddfSDavid du Colombier 	uint p;
67*7dd7cddfSDavid du Colombier 	char *ptr;
68*7dd7cddfSDavid du Colombier 
69*7dd7cddfSDavid du Colombier 	p = 0;
70*7dd7cddfSDavid du Colombier 	b = f->box;
71*7dd7cddfSDavid du Colombier 	trim = 0;
72*7dd7cddfSDavid du Colombier 	for(nb=0; nb<f->nbox && p<p1; nb++){
73*7dd7cddfSDavid du Colombier 		nr = b->nrune;
74*7dd7cddfSDavid du Colombier 		if(nr < 0)
75*7dd7cddfSDavid du Colombier 			nr = 1;
76*7dd7cddfSDavid du Colombier 		if(p+nr <= p0)
77*7dd7cddfSDavid du Colombier 			goto Continue;
78*7dd7cddfSDavid du Colombier 		if(p >= p0){
79*7dd7cddfSDavid du Colombier 			qt = pt;
80*7dd7cddfSDavid du Colombier 			_frcklinewrap(f, &pt, b);
81*7dd7cddfSDavid du Colombier 			if(pt.y > qt.y)
82*7dd7cddfSDavid du Colombier 				draw(f->b, Rect(qt.x, qt.y, f->r.max.x, pt.y), back, nil, qt);
83*7dd7cddfSDavid du Colombier 		}
84*7dd7cddfSDavid du Colombier 		ptr = (char*)b->ptr;
85*7dd7cddfSDavid du Colombier 		if(p < p0){	/* beginning of region: advance into box */
86*7dd7cddfSDavid du Colombier 			ptr += nbytes(ptr, p0-p);
87*7dd7cddfSDavid du Colombier 			nr -= (p0-p);
88*7dd7cddfSDavid du Colombier 			p = p0;
89*7dd7cddfSDavid du Colombier 		}
90*7dd7cddfSDavid du Colombier 		trim = 0;
91*7dd7cddfSDavid du Colombier 		if(p+nr > p1){	/* end of region: trim box */
92*7dd7cddfSDavid du Colombier 			nr -= (p+nr)-p1;
93*7dd7cddfSDavid du Colombier 			trim = 1;
94*7dd7cddfSDavid du Colombier 		}
95*7dd7cddfSDavid du Colombier 		if(b->nrune<0 || nr==b->nrune)
96*7dd7cddfSDavid du Colombier 			w = b->wid;
97*7dd7cddfSDavid du Colombier 		else
98*7dd7cddfSDavid du Colombier 			w = stringnwidth(f->font, ptr, nr);
99*7dd7cddfSDavid du Colombier 		x = pt.x+w;
100*7dd7cddfSDavid du Colombier 		if(x > f->r.max.x)
101*7dd7cddfSDavid du Colombier 			x = f->r.max.x;
102*7dd7cddfSDavid du Colombier 		draw(f->b, Rect(pt.x, pt.y, x, pt.y+f->font->height), back, nil, pt);
103*7dd7cddfSDavid du Colombier 		if(b->nrune >= 0)
104*7dd7cddfSDavid du Colombier 			stringn(f->b, pt, text, ZP, f->font, ptr, nr);
105*7dd7cddfSDavid du Colombier 		pt.x += w;
106*7dd7cddfSDavid du Colombier 	    Continue:
107*7dd7cddfSDavid du Colombier 		b++;
108*7dd7cddfSDavid du Colombier 		p += nr;
109*7dd7cddfSDavid du Colombier 	}
110*7dd7cddfSDavid du Colombier 	/* if this is end of last plain text box on wrapped line, fill to end of line */
111*7dd7cddfSDavid du Colombier 	if(p1>p0 &&  b>f->box && b<f->box+f->nbox && b[-1].nrune>0 && !trim){
112*7dd7cddfSDavid du Colombier 		qt = pt;
113*7dd7cddfSDavid du Colombier 		_frcklinewrap(f, &pt, b);
114*7dd7cddfSDavid du Colombier 		if(pt.y > qt.y)
115*7dd7cddfSDavid du Colombier 			draw(f->b, Rect(qt.x, qt.y, f->r.max.x, pt.y), back, nil, qt);
116*7dd7cddfSDavid du Colombier 	}
117*7dd7cddfSDavid du Colombier }
118*7dd7cddfSDavid du Colombier 
119*7dd7cddfSDavid du Colombier void
120*7dd7cddfSDavid du Colombier frtick(Frame *f, Point pt, int ticked)
121*7dd7cddfSDavid du Colombier {
122*7dd7cddfSDavid du Colombier 	Rectangle r;
123*7dd7cddfSDavid du Colombier 
124*7dd7cddfSDavid du Colombier 	if(f->ticked==ticked || f->tick==0 || !ptinrect(pt, f->r))
125*7dd7cddfSDavid du Colombier 		return;
126*7dd7cddfSDavid du Colombier 	pt.x--;	/* looks best just left of where requested */
127*7dd7cddfSDavid du Colombier 	r = Rect(pt.x, pt.y, pt.x+FRTICKW, pt.y+f->font->height);
128*7dd7cddfSDavid du Colombier 	if(ticked){
129*7dd7cddfSDavid du Colombier 		draw(f->tickback, f->tickback->r, f->b, nil, pt);
130*7dd7cddfSDavid du Colombier 		draw(f->b, r, f->tick, nil, ZP);
131*7dd7cddfSDavid du Colombier 	}else
132*7dd7cddfSDavid du Colombier 		draw(f->b, r, f->tickback, nil, ZP);
133*7dd7cddfSDavid du Colombier 	f->ticked = ticked;
134*7dd7cddfSDavid du Colombier }
135*7dd7cddfSDavid du Colombier 
1363e12c5d1SDavid du Colombier Point
1373e12c5d1SDavid du Colombier _frdraw(Frame *f, Point pt)
1383e12c5d1SDavid du Colombier {
1393e12c5d1SDavid du Colombier 	Frbox *b;
1403e12c5d1SDavid du Colombier 	int nb, n;
1413e12c5d1SDavid du Colombier 
1423e12c5d1SDavid du Colombier 	for(b=f->box,nb=0; nb<f->nbox; nb++, b++){
1433e12c5d1SDavid du Colombier 		_frcklinewrap0(f, &pt, b);
1443e12c5d1SDavid du Colombier 		if(pt.y == f->r.max.y){
1453e12c5d1SDavid du Colombier 			f->nchars -= _frstrlen(f, nb);
1463e12c5d1SDavid du Colombier 			_frdelbox(f, nb, f->nbox-1);
1473e12c5d1SDavid du Colombier 			break;
1483e12c5d1SDavid du Colombier 		}
1493e12c5d1SDavid du Colombier 		if(b->nrune > 0){
1503e12c5d1SDavid du Colombier 			n = _frcanfit(f, pt, b);
1513e12c5d1SDavid du Colombier 			if(n == 0)
152*7dd7cddfSDavid du Colombier 				drawerror(f->display, "draw: _frcanfit==0");
1533e12c5d1SDavid du Colombier 			if(n != b->nrune){
1543e12c5d1SDavid du Colombier 				_frsplitbox(f, nb, n);
1553e12c5d1SDavid du Colombier 				b = &f->box[nb];
1563e12c5d1SDavid du Colombier 			}
1573e12c5d1SDavid du Colombier 			pt.x += b->wid;
1583e12c5d1SDavid du Colombier 		}else{
159*7dd7cddfSDavid du Colombier 			if(b->bc == '\n'){
160*7dd7cddfSDavid du Colombier 				pt.x = f->r.min.x;
161*7dd7cddfSDavid du Colombier 				pt.y+=f->font->height;
162*7dd7cddfSDavid du Colombier 			}else
1633e12c5d1SDavid du Colombier 				pt.x += _frnewwid(f, pt, b);
1643e12c5d1SDavid du Colombier 		}
1653e12c5d1SDavid du Colombier 	}
1663e12c5d1SDavid du Colombier 	return pt;
1673e12c5d1SDavid du Colombier }
168*7dd7cddfSDavid du Colombier 
1693e12c5d1SDavid du Colombier int
1703e12c5d1SDavid du Colombier _frstrlen(Frame *f, int nb)
1713e12c5d1SDavid du Colombier {
1723e12c5d1SDavid du Colombier 	int n;
1733e12c5d1SDavid du Colombier 
1743e12c5d1SDavid du Colombier 	for(n=0; nb<f->nbox; nb++)
1753e12c5d1SDavid du Colombier 		n += NRUNE(&f->box[nb]);
1763e12c5d1SDavid du Colombier 	return n;
1773e12c5d1SDavid du Colombier }
178