ref: ddc6c1651574ca04fa73984452cdcc3dddb08256
parent: 34f5a8333090c5090e1d0b6ddbaf900d64340eea
author: cinap_lenrek <cinap_lenrek@felloff.net>
date: Mon Apr 17 01:12:07 EDT 2017
vt: implement snarf support
--- a/sys/src/cmd/vt/cons.h
+++ b/sys/src/cmd/vt/cons.h
@@ -8,9 +8,7 @@
extern Consstate* consctl(void);
extern Consstate* cs;
-#define XMARGIN 5 /* inset from border of layer */
-#define YMARGIN 5
-#define INSET 3
+#define INSET 2
#define BUFS 32
#define HISTSIZ 4096 /* number of history characters */
#define BSIZE 1000
@@ -30,8 +28,8 @@
TReverse = (1<<3),
TInvisible = (1<<4),
};
-
+#define button1() ((mouse.buttons & 07)==1)
#define button2() ((mouse.buttons & 07)==2)
#define button3() ((mouse.buttons & 07)==4)
@@ -55,7 +53,7 @@
extern void emulate(void);
extern int host_avail(void);
-extern void clear(Rectangle);
+extern void clear(int,int,int,int);
extern void newline(void);
extern int get_next_char(void);
extern void ringbell(void);
@@ -64,8 +62,9 @@
extern void backup(int);
extern void sendnchars(int, char *);
extern Point pt(int, int);
+extern Point pos(Point);
extern void funckey(int);
-extern void drawstring(Point, Rune*, int);
+extern void drawstring(Rune*, int, int);
extern int debug;
extern int yscrmin, yscrmax;
--- a/sys/src/cmd/vt/main.c
+++ b/sys/src/cmd/vt/main.c
@@ -47,6 +47,7 @@
Menu menu3;
Rune *histp;
Rune hist[HISTSIZ];
+Rune *onscreen;
int yscrmin, yscrmax;
int attr, defattr;
int wctlout;
@@ -56,10 +57,12 @@
Image *colors[8];
Image *hicolors[8];
Image *red;
+Image *green;
Image *fgcolor;
Image *bgcolor;
Image *fgdefault;
Image *bgdefault;
+Image *highlight;
uint rgbacolors[8] = {
0x000000FF, /* black */
@@ -88,6 +91,8 @@
int NS;
int CW;
+int XMARGIN;
+int YMARGIN;
Consstate *cs;
Mouse mouse;
@@ -119,11 +124,13 @@
void set_host(Event *);
void bigscroll(void);
void readmenu(void);
+void selection(void);
void eresized(int);
void resize(void);
void send_interrupt(void);
int alnum(int);
void escapedump(int,uchar *,int);
+Rune* onscreenp(int, int);
void
main(int argc, char **argv)
@@ -210,8 +217,10 @@
CW = stringwidth(font, "m");
red = allocimage(display, Rect(0,0,1,1), screen->chan, 1, DRed);
+ green = allocimage(display, Rect(0,0,1,1), screen->chan, 1, DGreen);
bordercol = allocimage(display, Rect(0,0,1,1), screen->chan, 1, 0xCCCCCCCC);
cursback = allocimage(display, Rect(0, 0, CW+1, NS+1), screen->chan, 0, DNofill);
+ highlight = allocimage(display, Rect(0,0,1,1), CHAN1(CAlpha,8), 1, 0x80);
for(i=0; i<8; i++){
colors[i] = allocimage(display, Rect(0,0,1,1), screen->chan, 1,
@@ -234,14 +243,23 @@
}
void
-clear(Rectangle r)
+clear(int x1, int y1, int x2, int y2)
{
- draw(screen, r, bgcolor, nil, ZP);
+ draw(screen, Rpt(pt(x1,y1), pt(x2,y2)), bgcolor, nil, ZP);
+ while(y1 < y2){
+ if(x1 < x2)
+ memset(onscreenp(x1, y1), 0, (x2-x1)*sizeof(Rune));
+ if(x2 > xmax)
+ *onscreenp(xmax+1, y1) = '\n';
+ y1++;
+ }
}
void
newline(void)
{
+ if(x > xmax)
+ *onscreenp(xmax+1, y) = 0; /* wrap arround, remove hidden newline */
nbacklines--;
if(y >= yscrmax) {
y = yscrmax;
@@ -427,8 +445,12 @@
wasblocked = blocked;
if(backp)
return(0);
- if(ecanmouse() && (button2() || button3()))
- readmenu();
+ if(ecanmouse()){
+ if(button1())
+ selection();
+ else if(button2() || button3())
+ readmenu();
+ }
if(snarffp) {
static Rune lastc = ~0;
@@ -535,6 +557,7 @@
continue;
}
curson(wasblocked); /* turn on cursor while we're waiting */
+ flushimage(display, 1);
do {
newmouse = 0;
switch(eread(blocked ? Emouse|Ekeyboard :
@@ -541,7 +564,9 @@
Emouse|Ekeyboard|Ehost, &e)) {
case Emouse:
mouse = e.mouse;
- if(button2() || button3())
+ if(button1())
+ selection();
+ else if(button2() || button3())
readmenu();
else if(resize_flag == 0) {
/* eresized() is triggered by special mouse event */
@@ -581,8 +606,8 @@
void
exportsize(void)
{
- putenvint("XPIXELS", Dx(screen->r)-2*XMARGIN);
- putenvint("YPIXELS", Dy(screen->r)-2*XMARGIN);
+ putenvint("XPIXELS", (xmax+1)*CW);
+ putenvint("YPIXELS", (ymax+1)*NS);
putenvint("LINES", ymax+1);
putenvint("COLS", xmax+1);
putenv("TERM", term);
@@ -595,17 +620,20 @@
fprint(2, "can't reattach to window: %r\n");
exits("can't reattach to window");
}
- xmax = (Dx(screen->r)-2*XMARGIN)/CW-1;
- ymax = (Dy(screen->r)-2*YMARGIN)/NS-1;
- if(xmax == 0 || ymax == 0)
- exits("window gone");
+ draw(screen, screen->r, bgcolor, nil, ZP);
+ xmax = (Dx(screen->r) - 2*INSET)/CW-1;
+ ymax = (Dy(screen->r) - 2*INSET)/NS-1;
+ XMARGIN = (Dx(screen->r) - (xmax+1)*CW) / 2;
+ YMARGIN = (Dy(screen->r) - (ymax+1)*NS) / 2;
x = 0;
y = 0;
yscrmin = 0;
yscrmax = ymax;
+ free(onscreen);
+ onscreen = mallocz((ymax+1)*(xmax+2)*sizeof(Rune), 1);
olines = 0;
exportsize();
- clear(screen->r);
+ clear(0,0,xmax+1,ymax+1);
if(resize_flag > 1)
backup(backc);
resize_flag = 0;
@@ -622,22 +650,103 @@
ymax = ht-1;
if(wid != -1)
xmax = wid-1;
-
r.min = screen->r.min;
- r.max = addpt(screen->r.min,
- Pt((xmax+1)*CW+2*XMARGIN+2*INSET,
- (ymax+1)*NS+2*YMARGIN+2*INSET));
+ r.max = addpt(screen->r.min, Pt((xmax+1)*CW+2*INSET, (ymax+1)*NS+2*INSET));
fd = open("/dev/wctl", OWRITE);
- if(fd < 0 || fprint(fd, "resize -dx %d -dy %d\n", Dx(r)+2*Borderwidth,
- Dy(r)+2*Borderwidth) < 0){
- border(screen, r, INSET, bordercol, ZP);
- exportsize();
- }
+ if(fd < 0 || fprint(fd, "resize -dx %d -dy %d\n", Dx(r)+2*Borderwidth, Dy(r)+2*Borderwidth) < 0)
+ resize();
if(fd >= 0)
close(fd);
}
void
+sendsnarf(void)
+{
+ if(snarffp == nil)
+ snarffp = Bopen("/dev/snarf",OREAD);
+}
+
+int
+writesnarf(Rune *s, Rune *e)
+{
+ Biobuf *b;
+ int z, p;
+
+ if(s >= e)
+ return 0;
+ b = Bopen("/dev/snarf", OWRITE|OTRUNC);
+ if(b == nil)
+ return 0;
+ for(z = p = 0; s < e; s++){
+ if(*s){
+ if(*s == '\n')
+ z = p = 0;
+ else if(p++ == 0){
+ while(z-- > 0) Bputc(b, ' ');
+ }
+ Bputrune(b, *s);
+ } else {
+ z++;
+ }
+ }
+ Bterm(b);
+ return 1;
+}
+
+Rectangle
+drawselection(Rectangle r, Rectangle d, Image *color)
+{
+ while(r.min.y < r.max.y){
+ d = drawselection(Rect(r.min.x, r.min.y, xmax+1, r.min.y), d, color);
+ r.min.x = 0;
+ r.min.y++;
+ }
+ if(r.min.x >= r.max.x)
+ return d;
+ r = Rpt(pt(r.min.x, r.min.y), pt(r.max.x, r.max.y+1));
+ draw(screen, r, color, highlight, r.min);
+ combinerect(&d, r);
+ return d;
+}
+
+void
+selection(void)
+{
+ Point p, q;
+ Rectangle r, d;
+ Image *backup;
+
+ backup = allocimage(display, screen->r, screen->chan, 0, DNofill);
+ draw(backup, backup->r, screen, nil, backup->r.min);
+ p = pos(mouse.xy);
+ do {
+ q = pos(mouse.xy);
+ if(onscreenp(p.x, p.y) > onscreenp(q.x, q.y)){
+ r.min = q;
+ r.max = p;
+ } else {
+ r.min = p;
+ r.max = q;
+ }
+ if(r.max.y > ymax)
+ r.max.x = 0;
+ d = drawselection(r, ZR, red);
+ flushimage(display, 1);
+ mouse = emouse();
+ draw(screen, d, backup, nil, d.min);
+ } while(button1());
+ if((mouse.buttons & 07) == 5)
+ sendsnarf();
+ else if(writesnarf(onscreenp(r.min.x, r.min.y), onscreenp(r.max.x, r.max.y))){
+ d = drawselection(r, ZR, green);
+ flushimage(display, 1);
+ sleep(200);
+ draw(screen, d, backup, nil, d.min);
+ }
+ freeimage(backup);
+}
+
+void
readmenu(void)
{
if(button3()) {
@@ -693,7 +802,7 @@
return;
case 4: /* send the snarf buffer */
- snarffp = Bopen("/dev/snarf",OREAD);
+ sendsnarf();
return;
case 5: /* pause and clear at end of screen */
@@ -745,12 +854,36 @@
return addpt(screen->r.min, Pt(x*CW+XMARGIN,y*NS+YMARGIN));
}
+Point
+pos(Point pt)
+{
+ pt.x -= screen->r.min.x + XMARGIN;
+ pt.y -= screen->r.min.y + YMARGIN;
+ pt.x /= CW;
+ pt.y /= NS;
+ if(pt.x < 0)
+ pt.x = 0;
+ else if(pt.x > xmax+1)
+ pt.x = xmax+1;
+ if(pt.y < 0)
+ pt.y = 0;
+ else if(pt.y > ymax+1)
+ pt.y = ymax+1;
+ return pt;
+}
+
+Rune*
+onscreenp(int x, int y)
+{
+ return onscreen + (y*(xmax+2) + x);
+}
+
void
scroll(int sy, int ly, int dy, int cy) /* source, limit, dest, which line to clear */
{
+ memmove(onscreenp(0, dy), onscreenp(0, sy), (ly-sy)*(xmax+2)*sizeof(Rune));
draw(screen, Rpt(pt(0, dy), pt(xmax+1, dy+ly-sy)), screen, nil, pt(0, sy));
- clear(Rpt(pt(0, cy), pt(xmax+1, cy+1)));
- flushimage(display, 1);
+ clear(0, cy, xmax+1, cy+1);
}
void
@@ -761,16 +894,16 @@
if(x == 0 && y == 0)
return;
if(y < half) {
- clear(Rpt(pt(0,0),pt(xmax+1,ymax+1)));
+ clear(0, 0, xmax+1, ymax+1);
x = y = 0;
return;
}
draw(screen, Rpt(pt(0, 0), pt(xmax+1, ymax+1)), screen, nil, pt(0, half));
- clear(Rpt(pt(0,y-half+1),pt(xmax+1,ymax+1)));
+ memmove(onscreenp(0, 0), onscreenp(0, half), (ymax-half+1)*(xmax+2)*sizeof(Rune));
+ clear(0, y-half+1, xmax+1, ymax+1);
y -= half;
if(olines)
olines -= half;
- flushimage(display, 1);
}
int
@@ -888,11 +1021,12 @@
void
-drawstring(Point p, Rune *str, int attr)
+drawstring(Rune *str, int n, int attr)
{
int i;
Image *txt, *bg, *tmp;
-
+ Point p;
+
txt = fgcolor;
bg = bgcolor;
if(attr & TReverse){
@@ -905,7 +1039,8 @@
if(txt == colors[i])
txt = hicolors[i];
}
-
+ p = pt(x, y);
draw(screen, Rpt(p, addpt(p, runestringsize(font, str))), bg, nil, p);
runestring(screen, p, txt, ZP, font, str);
+ memmove(onscreenp(x, y), str, n*sizeof(Rune));
}
--- a/sys/src/cmd/vt/vt.c
+++ b/sys/src/cmd/vt/vt.c
@@ -687,21 +687,21 @@
* operand 2: whole screen.
*/
case 2:
- clear(Rpt(pt(0, 0), pt(xmax+1, ymax+1)));
+ clear(0, 0, xmax+1, ymax+1);
break;
/*
* operand 1: start of screen to active position, inclusive.
*/
case 1:
- clear(Rpt(pt(0, 0), pt(xmax+1, y)));
- clear(Rpt(pt(0, y), pt(x+1, y+1)));
+ clear(0, 0, xmax+1, y);
+ clear(0, y, x+1, y+1);
break;
/*
* Default: active position to end of screen, inclusive.
*/
default:
- clear(Rpt(pt(x, y), pt(xmax+1, y+1)));
- clear(Rpt(pt(0, y+1), pt(xmax+1, ymax+1)));
+ clear(x, y, xmax+1, y+1);
+ clear(0, y+1, xmax+1, ymax+1);
break;
}
break;
@@ -715,19 +715,19 @@
* operand 2: whole line.
*/
case 2:
- clear(Rpt(pt(0, y), pt(xmax+1, y+1)));
+ clear(0, y, xmax+1, y+1);
break;
/*
* operand 1: start of line to active position, inclusive.
*/
case 1:
- clear(Rpt(pt(0, y), pt(x+1, y+1)));
+ clear(0, y, x+1, y+1);
break;
/*
* Default: active position to end of line, inclusive.
*/
default:
- clear(Rpt(pt(x, y), pt(xmax+1, y+1)));
+ clear(x, y, xmax+1, y+1);
break;
}
break;
@@ -739,7 +739,7 @@
fixops(operand);
i = x + operand[0];
draw(screen, Rpt(pt(x, y), pt(xmax+1, y+1)), screen, nil, pt(i, y));
- clear(Rpt(pt(xmax-operand[0], y), pt(xmax+1, y+1)));
+ clear(xmax-operand[0], y, xmax+1, y+1);
break;
/*
@@ -749,7 +749,7 @@
fixops(operand);
i = x + operand[0];
draw(screen, Rpt(pt(i, y), pt(xmax+1, y+1)), screen, nil, pt(x, y));
- clear(Rpt(pt(x, y), pt(i, y+1)));
+ clear(x, y, i, y+1);
break;
@@ -759,7 +759,7 @@
case 'X':
fixops(operand);
i = x + operand[0];
- clear(Rpt(pt(x, y), pt(i, y+1)));
+ clear(x, y, i, y+1);
break;
/*
@@ -844,8 +844,8 @@
/* line wrap */
if (x > xmax){
if(wraparound){
- x = 0;
newline();
+ x = 0;
}else{
continue;
}
@@ -858,7 +858,7 @@
c = 0;
}
buf[n] = 0;
- drawstring(pt(x, y), buf, attr);
+ drawstring(buf, n, attr);
x += n;
peekc = c;
break;