14 static Rune Lheader[] = {
17 'P', 'a', 's', 't', 'e', ' ',
18 'S', 'n', 'a', 'r', 'f', ' ',
19 'S', 'o', 'r', 't', ' ',
20 'Z', 'e', 'r', 'o', 'x', ' ',
21 'D', 'e', 'l', 'c', 'o', 'l', ' ',
26 colinit(Column *c, Rectangle r)
31 draw(screen, r, display->white, nil, ZP);
39 r1.max.y = r1.min.y + font->height;
40 textinit(t, fileaddtext(nil, t), r1, &reffont, tagcols);
44 draw(screen, r1, display->black, nil, ZP);
45 textinsert(t, 0, Lheader, 38, TRUE);
46 textsetselect(t, t->file->b.nc, t->file->b.nc);
47 draw(screen, t->scrollr, colbutton, nil, colbutton->r.min);
52 coladd(Column *c, Window *w, Window *clone, int y)
60 r.min.y = c->tag.fr.r.max.y+Border;
61 if(y<r.min.y && c->nw>0){ /* steal half of last window by default */
63 y = v->body.fr.r.min.y+Dy(v->body.fr.r)/2;
65 /* look for window we'll land on */
66 for(i=0; i<c->nw; i++){
73 i++; /* new window will go after v */
75 * if v's too small, grow it first.
77 if(!c->safe || v->body.fr.maxlines<=3){
79 y = v->body.fr.r.min.y+Dy(v->body.fr.r)/2;
85 t = c->w[i]->r.min.y-Border;
87 draw(screen, r, textcols[BACK], nil, ZP);
89 y = min(y, t-(v->tag.fr.font->height+v->body.fr.font->height+Border+1));
90 r1.max.y = min(y, v->body.fr.r.min.y+v->body.fr.nlines*v->body.fr.font->height);
91 r1.min.y = winresize(v, r1, FALSE);
92 r1.max.y = r1.min.y+Border;
93 draw(screen, r1, display->black, nil, ZP);
97 w = emalloc(sizeof(Window));
99 draw(screen, r, textcols[BACK], nil, ZP);
100 wininit(w, clone, r);
103 winresize(w, r, FALSE);
108 w->body.row = c->row;
109 c->w = realloc(c->w, (c->nw+1)*sizeof(Window*));
110 memmove(c->w+i+1, c->w+i, (c->nw-i)*sizeof(Window*));
114 /* near but not on the button */
115 moveto(mousectl, addpt(w->tag.scrollr.max, Pt(3, 3)));
122 colclose(Column *c, Window *w, int dofree)
130 for(i=0; i<c->nw; i++)
133 error("can't find window");
144 memmove(c->w+i, c->w+i+1, (c->nw-i)*sizeof(Window*));
146 c->w = realloc(c->w, c->nw*sizeof(Window*));
148 draw(screen, r, display->white, nil, ZP);
151 if(i == c->nw){ /* extend last window down */
153 r.min.y = w->r.min.y;
154 r.max.y = c->r.max.y;
155 }else{ /* extend next window up */
157 r.max.y = w->r.max.y;
159 draw(screen, r, textcols[BACK], nil, ZP);
161 winresize(w, r, FALSE);
165 colcloseall(Column *c)
173 for(i=0; i<c->nw; i++){
184 colmousebut(Column *c)
186 moveto(mousectl, divpt(addpt(c->tag.scrollr.min, c->tag.scrollr.max), 2));
190 colresize(Column *c, Rectangle r)
198 r1.max.y = r1.min.y + c->tag.fr.font->height;
199 textresize(&c->tag, r1);
200 draw(screen, c->tag.scrollr, colbutton, nil, colbutton->r.min);
203 draw(screen, r1, display->black, nil, ZP);
205 for(i=0; i<c->nw; i++){
211 r1.max.y = r1.min.y+(Dy(w->r)+Border)*Dy(r)/Dy(c->r);
213 r2.max.y = r2.min.y+Border;
214 draw(screen, r2, display->black, nil, ZP);
216 r1.min.y = winresize(w, r1, FALSE);
223 colcmp(const void *a, const void *b)
228 r1 = (*(Window**)a)->body.file->name;
229 nr1 = (*(Window**)a)->body.file->nname;
230 r2 = (*(Window**)b)->body.file->name;
231 nr2 = (*(Window**)b)->body.file->nname;
232 for(i=0; i<nr1 && i<nr2; i++){
245 Rectangle r, r1, *rp;
251 rp = emalloc(c->nw*sizeof(Rectangle));
252 wp = emalloc(c->nw*sizeof(Window*));
253 memmove(wp, c->w, c->nw*sizeof(Window*));
254 qsort(wp, c->nw, sizeof(Window*), colcmp);
255 for(i=0; i<c->nw; i++)
258 r.min.y = c->tag.fr.r.max.y;
259 draw(screen, r, textcols[BACK], nil, ZP);
261 for(i=0; i<c->nw; i++){
265 r.max.y = c->r.max.y;
267 r.max.y = r.min.y+Dy(w->r)+Border;
269 r1.max.y = r1.min.y+Border;
270 draw(screen, r1, display->black, nil, ZP);
272 y = winresize(w, r, FALSE);
280 colgrow(Column *c, Window *w, int but)
283 int i, j, k, l, y1, y2, *nl, *ny, tot, nnl, onl, dnl, h;
286 for(i=0; i<c->nw; i++)
289 error("can't find window");
293 if(but < 0){ /* make sure window fills its own space properly */
295 if(i==c->nw-1 || c->safe==FALSE)
298 r.max.y = c->w[i+1]->r.min.y;
299 winresize(w, r, FALSE);
302 cr.min.y = c->w[0]->r.min.y;
303 if(but == 3){ /* full size */
309 draw(screen, cr, textcols[BACK], nil, ZP);
310 winresize(w, cr, FALSE);
311 for(i=1; i<c->nw; i++)
312 c->w[i]->body.fr.maxlines = 0;
316 /* store old #lines for each window */
317 onl = w->body.fr.maxlines;
318 nl = emalloc(c->nw * sizeof(int));
319 ny = emalloc(c->nw * sizeof(int));
321 for(j=0; j<c->nw; j++){
322 l = c->w[j]->body.fr.maxlines;
326 /* approximate new #lines for this window */
327 if(but == 2){ /* as big as can be */
328 memset(nl, 0, c->nw * sizeof(int));
331 nnl = min(onl + max(min(5, w->maxlines), onl/2), tot);
332 if(nnl < w->maxlines)
333 nnl = (w->maxlines+nnl)/2;
337 /* compute new #lines for each window */
338 for(k=1; k<c->nw; k++){
339 /* prune from later window */
341 if(j<c->nw && nl[j]){
342 l = min(dnl, max(1, nl[j]/2));
347 /* prune from earlier window */
350 l = min(dnl, max(1, nl[j]/2));
357 /* pack everyone above */
363 r.max.y = y1+Dy(v->tag.all);
365 r.max.y += 1 + nl[j]*v->body.fr.font->height;
366 if(!c->safe || !eqrect(v->r, r)){
367 draw(screen, r, textcols[BACK], nil, ZP);
368 winresize(v, r, c->safe);
370 r.min.y = v->r.max.y;
372 draw(screen, r, display->black, nil, ZP);
375 /* scan to see new size of everyone below */
377 for(j=c->nw-1; j>i; j--){
380 r.min.y = y2-Dy(v->tag.all);
382 r.min.y -= 1 + nl[j]*v->body.fr.font->height;
387 /* compute new size of window */
390 r.max.y = r.min.y+Dy(w->tag.all);
391 h = w->body.fr.font->height;
392 if(y2-r.max.y >= 1+h+Border){
394 r.max.y += h*((y2-r.max.y)/h);
397 if(!c->safe || !eqrect(w->r, r)){
398 draw(screen, r, textcols[BACK], nil, ZP);
399 winresize(w, r, c->safe);
404 draw(screen, r, display->black, nil, ZP);
405 for(j=i+1; j<c->nw; j++)
406 ny[j] -= (y2-r.max.y);
408 /* pack everyone below */
410 for(j=i+1; j<c->nw; j++){
414 r.max.y = y1+Dy(v->tag.all);
416 r.max.y += 1 + nl[j]*v->body.fr.font->height;
417 if(!c->safe || !eqrect(v->r, r)){
418 draw(screen, r, textcols[BACK], nil, ZP);
419 winresize(v, r, c->safe);
421 if(j < c->nw-1){ /* no border on last window */
422 r.min.y = v->r.max.y;
424 draw(screen, r, display->black, nil, ZP);
430 r.max.y = c->r.max.y;
431 draw(screen, r, textcols[BACK], nil, ZP);
439 coldragwin(Column *c, Window *w, int but)
448 setcursor(mousectl, &boxcursor);
451 while(mouse->buttons == b)
453 setcursor(mousectl, nil);
455 while(mouse->buttons)
460 for(i=0; i<c->nw; i++)
463 error("can't find window");
467 if(abs(p.x-op.x)<5 && abs(p.y-op.y)<5){
472 /* is it a flick to the right? */
473 if(abs(p.y-op.y)<10 && p.x>op.x+30 && rowwhichcol(c->row, p)==c)
474 p.x += Dx(w->r); /* yes: toss to next column */
475 nc = rowwhichcol(c->row, p);
476 if(nc!=nil && nc!=c){
477 colclose(c, w, FALSE);
478 coladd(nc, w, nil, p.y);
483 return; /* can't do it */
484 if((i>0 && p.y<c->w[i-1]->r.min.y) || (i<c->nw-1 && p.y>w->r.max.y)
485 || (i==0 && p.y>w->r.max.y)){
487 colclose(c, w, FALSE);
488 coladd(c, w, nil, p.y);
495 if(p.y < v->tag.all.max.y)
496 p.y = v->tag.all.max.y;
497 if(p.y > w->r.max.y-Dy(w->tag.all)-Border)
498 p.y = w->r.max.y-Dy(w->tag.all)-Border;
501 if(r.max.y > v->body.fr.r.min.y){
502 r.max.y -= (r.max.y-v->body.fr.r.min.y)%v->body.fr.font->height;
503 if(v->body.fr.r.min.y == v->body.fr.r.max.y)
506 if(!eqrect(v->r, r)){
507 draw(screen, r, textcols[BACK], nil, ZP);
508 winresize(v, r, c->safe);
510 r.min.y = v->r.max.y;
511 r.max.y = r.min.y+Border;
512 draw(screen, r, display->black, nil, ZP);
515 r.max.y = c->r.max.y;
517 r.max.y = c->w[i+1]->r.min.y-Border;
518 if(!eqrect(w->r, r)){
519 draw(screen, r, textcols[BACK], nil, ZP);
520 winresize(w, r, c->safe);
527 colwhich(Column *c, Point p)
532 if(!ptinrect(p, c->r))
534 if(ptinrect(p, c->tag.all))
536 for(i=0; i<c->nw; i++){
538 if(ptinrect(p, w->r)){
539 if(ptinrect(p, w->tag.all))
553 for(i=0; i<c->nw; i++)
554 clean &= winclean(c->w[i], TRUE);