15 static Rune Lheader[] = {
18 'P', 'a', 's', 't', 'e', ' ',
19 'S', 'n', 'a', 'r', 'f', ' ',
20 'S', 'o', 'r', 't', ' ',
21 'Z', 'e', 'r', 'o', 'x', ' ',
22 'D', 'e', 'l', 'c', 'o', 'l', ' ',
27 colinit(Column *c, Rectangle r)
32 draw(screen, r, display->white, nil, ZP);
40 r1.max.y = r1.min.y + font->height;
41 textinit(t, fileaddtext(nil, t), r1, &reffont, tagcols);
45 draw(screen, r1, display->black, nil, ZP);
46 textinsert(t, 0, Lheader, 38, TRUE);
47 textsetselect(t, t->file->b.nc, t->file->b.nc);
48 draw(screen, t->scrollr, colbutton, nil, colbutton->r.min);
53 coladd(Column *c, Window *w, Window *clone, int y)
57 int i, j, minht, ymax, buggered;
61 r.min.y = c->tag.fr.r.max.y+Border;
62 if(y<r.min.y && c->nw>0){ /* steal half of last window by default */
64 y = v->body.fr.r.min.y+Dy(v->body.fr.r)/2;
66 /* look for window we'll land on */
67 for(i=0; i<c->nw; i++){
75 i++; /* new window will go after v */
77 * if landing window (v) is too small, grow it first.
79 minht = v->tag.fr.font->height+Border+1;
81 while(!c->safe || v->body.fr.maxlines<=3 || Dy(v->body.all) <= minht){
83 buggered = 1; /* too many windows in column */
90 * figure out where to split v to make room for w
93 /* new window stops where next window begins */
95 ymax = c->w[i]->r.min.y-Border;
99 /* new window must start after v's tag ends */
100 y = max(y, v->tagtop.max.y+Border);
102 /* new window must start early enough to end before ymax */
103 y = min(y, ymax - minht);
105 /* if y is too small, too many windows in column */
106 if(y < v->tagtop.max.y+Border)
114 draw(screen, r, textcols[BACK], nil, ZP);
116 y = min(y, ymax-(v->tag.fr.font->height*v->taglines+v->body.fr.font->height+Border+1));
117 r1.max.y = min(y, v->body.fr.r.min.y+v->body.fr.nlines*v->body.fr.font->height);
118 r1.min.y = winresize(v, r1, FALSE, FALSE);
119 r1.max.y = r1.min.y+Border;
120 draw(screen, r1, display->black, nil, ZP);
123 * leave r with w's coordinates
128 w = emalloc(sizeof(Window));
130 draw(screen, r, textcols[BACK], nil, ZP);
131 wininit(w, clone, r);
134 winresize(w, r, FALSE, TRUE);
139 w->body.row = c->row;
140 c->w = realloc(c->w, (c->nw+1)*sizeof(Window*));
141 memmove(c->w+i+1, c->w+i, (c->nw-i)*sizeof(Window*));
146 /* if there were too many windows, redraw the whole column */
151 /* near the button, but in the body */
152 moveto(mousectl, addpt(w->tag.scrollr.max, Pt(3, 3)));
158 colclose(Column *c, Window *w, int dofree)
166 for(i=0; i<c->nw; i++)
169 error("can't find window");
175 didmouse = restoremouse(w);
181 memmove(c->w+i, c->w+i+1, (c->nw-i)*sizeof(Window*));
182 c->w = realloc(c->w, c->nw*sizeof(Window*));
184 draw(screen, r, display->white, nil, ZP);
188 if(i == c->nw){ /* extend last window down */
190 r.min.y = w->r.min.y;
191 r.max.y = c->r.max.y;
192 }else{ /* extend next window up */
195 r.max.y = w->r.max.y;
197 draw(screen, r, textcols[BACK], nil, ZP);
201 winresize(w, r, FALSE, TRUE);
208 colcloseall(Column *c)
216 for(i=0; i<c->nw; i++){
227 colmousebut(Column *c)
229 moveto(mousectl, divpt(addpt(c->tag.scrollr.min, c->tag.scrollr.max), 2));
233 colresize(Column *c, Rectangle r)
241 r1.max.y = r1.min.y + c->tag.fr.font->height;
242 textresize(&c->tag, r1, TRUE);
243 draw(screen, c->tag.scrollr, colbutton, nil, colbutton->r.min);
246 draw(screen, r1, display->black, nil, ZP);
248 for(i=0; i<c->nw; i++){
256 r1.max.y += (Dy(w->r)+Border)*Dy(r)/Dy(c->r);
259 r1.max.y = max(r1.max.y, r1.min.y + Border+font->height);
261 r2.max.y = r2.min.y+Border;
262 draw(screen, r2, display->black, nil, ZP);
264 r1.min.y = winresize(w, r1, FALSE, i==c->nw-1);
271 colcmp(const void *a, const void *b)
276 r1 = (*(Window**)a)->body.file->name;
277 nr1 = (*(Window**)a)->body.file->nname;
278 r2 = (*(Window**)b)->body.file->name;
279 nr2 = (*(Window**)b)->body.file->nname;
280 for(i=0; i<nr1 && i<nr2; i++){
293 Rectangle r, r1, *rp;
299 rp = emalloc(c->nw*sizeof(Rectangle));
300 wp = emalloc(c->nw*sizeof(Window*));
301 memmove(wp, c->w, c->nw*sizeof(Window*));
302 qsort(wp, c->nw, sizeof(Window*), colcmp);
303 for(i=0; i<c->nw; i++)
306 r.min.y = c->tag.fr.r.max.y;
307 draw(screen, r, textcols[BACK], nil, ZP);
309 for(i=0; i<c->nw; i++){
313 r.max.y = c->r.max.y;
315 r.max.y = r.min.y+Dy(w->r)+Border;
317 r1.max.y = r1.min.y+Border;
318 draw(screen, r1, display->black, nil, ZP);
320 y = winresize(w, r, FALSE, i==c->nw-1);
328 colgrow(Column *c, Window *w, int but)
331 int i, j, k, l, y1, y2, *nl, *ny, tot, nnl, onl, dnl, h;
334 for(i=0; i<c->nw; i++)
337 error("can't find window");
341 if(but < 0){ /* make sure window fills its own space properly */
343 if(i==c->nw-1 || c->safe==FALSE)
346 r.max.y = c->w[i+1]->r.min.y - Border;
347 winresize(w, r, FALSE, TRUE);
350 cr.min.y = c->w[0]->r.min.y;
351 if(but == 3){ /* full size */
357 draw(screen, cr, textcols[BACK], nil, ZP);
358 winresize(w, cr, FALSE, TRUE);
359 for(i=1; i<c->nw; i++)
360 c->w[i]->body.fr.maxlines = 0;
364 /* store old #lines for each window */
365 onl = w->body.fr.maxlines;
366 nl = emalloc(c->nw * sizeof(int));
367 ny = emalloc(c->nw * sizeof(int));
369 for(j=0; j<c->nw; j++){
370 l = c->w[j]->taglines-1 + c->w[j]->body.fr.maxlines;
374 /* approximate new #lines for this window */
375 if(but == 2){ /* as big as can be */
376 memset(nl, 0, c->nw * sizeof(int));
379 nnl = min(onl + max(min(5, w->taglines-1+w->maxlines), onl/2), tot);
380 if(nnl < w->taglines-1+w->maxlines)
381 nnl = (w->taglines-1+w->maxlines + nnl)/2;
385 /* compute new #lines for each window */
386 for(k=1; k<c->nw; k++){
387 /* prune from later window */
389 if(j<c->nw && nl[j]){
390 l = min(dnl, max(1, nl[j]/2));
395 /* prune from earlier window */
398 l = min(dnl, max(1, nl[j]/2));
405 /* pack everyone above */
411 r.max.y = y1+Dy(v->tagtop);
413 r.max.y += 1 + nl[j]*v->body.fr.font->height;
414 r.min.y = winresize(v, r, c->safe, FALSE);
416 draw(screen, r, display->black, nil, ZP);
419 /* scan to see new size of everyone below */
421 for(j=c->nw-1; j>i; j--){
424 r.min.y = y2-Dy(v->tagtop);
426 r.min.y -= 1 + nl[j]*v->body.fr.font->height;
431 /* compute new size of window */
435 h = w->body.fr.font->height;
436 if(Dy(r) < Dy(w->tagtop)+1+h+Border)
437 r.max.y = r.min.y + Dy(w->tagtop)+1+h+Border;
439 r.max.y = winresize(w, r, c->safe, TRUE);
443 draw(screen, r, display->black, nil, ZP);
444 for(j=i+1; j<c->nw; j++)
445 ny[j] -= (y2-r.max.y);
447 /* pack everyone below */
449 for(j=i+1; j<c->nw; j++){
453 r.max.y = y1+Dy(v->tagtop);
455 r.max.y += 1 + nl[j]*v->body.fr.font->height;
456 y1 = winresize(v, r, c->safe, j==c->nw-1);
457 if(j < c->nw-1){ /* no border on last window */
460 draw(screen, r, display->black, nil, ZP);
471 coldragwin(Column *c, Window *w, int but)
480 setcursor2(mousectl, &boxcursor, &boxcursor2);
483 while(mouse->buttons == b)
485 setcursor(mousectl, nil);
487 while(mouse->buttons)
492 for(i=0; i<c->nw; i++)
495 error("can't find window");
498 if(w->tagexpand) /* force recomputation of window tag size */
501 if(abs(p.x-op.x)<5 && abs(p.y-op.y)<5){
506 /* is it a flick to the right? */
507 if(abs(p.y-op.y)<10 && p.x>op.x+30 && rowwhichcol(c->row, p)==c)
508 p.x = op.x+Dx(w->r); /* yes: toss to next column */
509 nc = rowwhichcol(c->row, p);
510 if(nc!=nil && nc!=c){
511 colclose(c, w, FALSE);
512 coladd(nc, w, nil, p.y);
517 return; /* can't do it */
518 if((i>0 && p.y<c->w[i-1]->r.min.y) || (i<c->nw-1 && p.y>w->r.max.y)
519 || (i==0 && p.y>w->r.max.y)){
521 colclose(c, w, FALSE);
522 coladd(c, w, nil, p.y);
529 if(p.y < v->tagtop.max.y)
530 p.y = v->tagtop.max.y;
531 if(p.y > w->r.max.y-Dy(w->tagtop)-Border)
532 p.y = w->r.max.y-Dy(w->tagtop)-Border;
535 if(r.max.y > v->body.fr.r.min.y){
536 r.max.y -= (r.max.y-v->body.fr.r.min.y)%v->body.fr.font->height;
537 if(v->body.fr.r.min.y == v->body.fr.r.max.y)
540 r.min.y = winresize(v, r, c->safe, FALSE);
541 r.max.y = r.min.y+Border;
542 draw(screen, r, display->black, nil, ZP);
545 r.max.y = c->r.max.y;
547 r.max.y = c->w[i+1]->r.min.y-Border;
548 winresize(w, r, c->safe, TRUE);
554 colwhich(Column *c, Point p)
559 if(!ptinrect(p, c->r))
561 if(ptinrect(p, c->tag.all))
563 for(i=0; i<c->nw; i++){
565 if(ptinrect(p, w->r)){
566 if(ptinrect(p, w->tagtop) || ptinrect(p, w->tag.all))
568 /* exclude partial line at bottom */
569 if(p.x >= w->body.scrollr.max.x && p.y >= w->body.fr.r.max.y)
583 for(i=0; i<c->nw; i++)
584 clean &= winclean(c->w[i], TRUE);