ref: 3362539850d86c68b403202d113c67f2cd858fca
dir: /vdiff-shed/
diff 3637e076560fdb3cb06ccd88ab08c530f931c558 uncommitted --- a/vdiff.c +++ b/vdiff.c @@ -39,7 +39,7 @@ Rectangle scrposr; Rectangle listr; Rectangle textr; -Image *cols[Ncols]; +Image *cols[Ncols], *bgcols[Ncols]; Image *scrollbg; int scrollsize; int lineh; @@ -48,28 +48,36 @@ Line **lines; int lsize; int lcount; +int maxlength; +int Δpan; const char ellipsis[] = "..."; void drawline(Rectangle r, Line *l) { - Image *bg; + int nc, tab, off; Point p; Rune rn; char *s; - bg = cols[l->t]; - draw(screen, r, bg, nil, ZP); + draw(screen, r, bgcols[l->t], nil, ZP); p = Pt(r.min.x + Hpadding, r.min.y + (Dy(r)-font->height)/2); - for(s = l->s; *s; s++){ - if(*s == '\t') - p = string(screen, p, display->black, ZP, font, " "); - else if((p.x+Hpadding+stringwidth(font, " ")+stringwidth(font, ellipsis)>=textr.max.x)){ - string(screen, p, display->black, ZP, font, ellipsis); + off = Δpan / stringwidth(font, " "); + for(s = l->s, nc = 0, tab = 0; *s; nc++, off--, tab--){ + if(*s == '\t'){ + tab = 4 - nc % 4; + s++; + } + if(tab > 0){ + if(off <= 0) + p = runestring(screen, p, bgcols[l->t], ZP, font, L"█"); + }else if((p.x+Hpadding+stringwidth(font, " ")+stringwidth(font, ellipsis)>=textr.max.x)){ + string(screen, p, display->white, ZP, font, ellipsis); break; }else{ - s += chartorune(&rn, s) - 1; - p = runestringn(screen, p, display->black, ZP, font, &rn, 1); + s += chartorune(&rn, s); + if(off <= 0) + p = runestringn(screen, p, cols[l->t], ZP, font, &rn, 1); } } } @@ -80,8 +88,8 @@ Rectangle lr; int i, h, y; - draw(screen, sr, display->white, nil, ZP); - draw(screen, scrollr, scrollbg, nil, ZP); + draw(screen, sr, display->black, nil, ZP); + draw(screen, scrollr, display->black, nil, ZP); if(lcount>0){ h = ((double)nlines/lcount)*Dy(scrollr); y = ((double)offset/lcount)*Dy(scrollr); @@ -88,7 +96,7 @@ scrposr = Rect(scrollr.min.x, scrollr.min.y+y, scrollr.max.x-1, scrollr.min.y+y+h); }else scrposr = Rect(scrollr.min.x, scrollr.min.y, scrollr.max.x-1, scrollr.max.y); - draw(screen, scrposr, display->white, nil, ZP); + draw(screen, scrposr, scrollbg, nil, ZP); for(i=0; i<nlines && offset+i<lcount; i++){ lr = Rect(textr.min.x, textr.min.y+i*lineh, textr.max.x, textr.min.y+(i+1)*lineh); drawline(lr, lines[offset+i]); @@ -96,6 +104,20 @@ } void +pan(int off) +{ + int max; + + max = Hpadding + maxlength * stringwidth(font, " ") + 2 * stringwidth(font, ellipsis) - Dx(textr); + Δpan += off * stringwidth(font, " "); + if(Δpan < 0 || max <= 0) + Δpan = 0; + else if(Δpan > max) + Δpan = max; + redraw(); +} + +void scroll(int off) { if(off<0 && offset<=0) @@ -136,8 +158,9 @@ textr = insetrect(listr, Margin); lineh = Vpadding+font->height+Vpadding; nlines = Dy(textr)/lineh; - offset = 0; scrollsize = mousescrollsize(nlines); + if(offset > 0 && offset+nlines>lcount) + offset = lcount-nlines+1; redraw(); } @@ -147,12 +170,17 @@ Rectangle cr; cr = Rect(0, 0, 1, 1); - cols[Lfile] = allocimage(display, cr, screen->chan, 1, 0xefefefff); - cols[Lsep] = allocimage(display, cr, screen->chan, 1, 0xeaffffff); - cols[Ladd] = allocimage(display, cr, screen->chan, 1, 0xe6ffedff); - cols[Ldel] = allocimage(display, cr, screen->chan, 1, 0xffeef0ff); - cols[Lnone] = display->white; - scrollbg = allocimage(display, cr, screen->chan, 1, 0x999999ff); + cols[Lfile] = allocimage(display, cr, screen->chan, 1, 0xE7DA36FF); + cols[Lsep] = allocimage(display, cr, screen->chan, 1, 0xBEBEBEFF); + cols[Ladd] = allocimage(display, cr, screen->chan, 1, 0x55DD55FF); + cols[Ldel] = allocimage(display, cr, screen->chan, 1, 0xDD5555FF); + cols[Lnone] = allocimage(display, cr, screen->chan, 1, 0x777777FF); + bgcols[Lfile] = allocimage(display, cr, screen->chan, 1, 0x222222FF); + bgcols[Lsep] = bgcols[Lfile]; + bgcols[Ladd] = allocimage(display, cr, screen->chan, 1, 0x061106FF); + bgcols[Ldel] = allocimage(display, cr, screen->chan, 1, 0x060606FF); + bgcols[Lnone] = display->black; + scrollbg = cols[Lnone]; } int @@ -184,6 +212,8 @@ sysfatal("malloc: %r"); l->t = linetype(s); l->s = s; + if(strlen(s) > maxlength) + maxlength = strlen(s); l->l = n; if(l->t != Lfile && l->t != Lsep) l->f = f; @@ -233,10 +263,13 @@ l = parseline(f, n, s); if(l->t == Lfile && l->s[0] == '-' && strncmp(l->s+4, "a/", 2)==0) ab = 1; - if(l->t == Lfile && l->s[0] == '+'){ + else if(l->t == Lfile && l->s[0] == '+'){ f = l->s+4; - if(ab && strncmp(f, "b/", 2)==0) + if(ab && strncmp(f, "b/", 2)==0){ f += 1; + if(access(f, AEXIST) < 0) + f += 1; + } t = strchr(f, '\t'); if(t!=nil) *t = 0; @@ -309,7 +342,7 @@ break; } - if(ev.mouse.buttons&4){ + if(ev.mouse.buttons&2){ n = indexat(ev.mouse.xy); if(n>=0 && lines[n+offset]->f != nil) plumb(lines[n+offset]->f, lines[n+offset]->l); @@ -335,6 +368,18 @@ break; case Kpgdown: scroll(nlines); + break; + case Kup: + scroll(-1); + break; + case Kdown: + scroll(1); + break; + case Kright: + pan(4); + break; + case Kleft: + pan(-4); break; } break;