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