ref: 2e9835e771c54cb9a0c9cd81012e7b4b75f878f7
parent: 217e8a06198cf6681fb0c38c26f34d96ae4302f6
author: cinap_lenrek <[email protected]>
date: Tue Sep 4 16:55:10 EDT 2018
vt: fix selection past baseline (thanks BurnZeZ)
--- a/sys/src/cmd/vt/main.c
+++ b/sys/src/cmd/vt/main.c
@@ -162,7 +162,7 @@
int rcvchar(void);
void bigscroll(void);
void readmenu(void);
-void selection(void);
+void selecting(void);
int selected(int, int);
void resized(void);
void drawcursor(void);
@@ -810,7 +810,7 @@
switch(alt(a)){
case AMOUSE:
if(button1())
- selection();
+ selecting();
else if(button2() || button3())
readmenu();
else if(resize_flag == 0)
@@ -912,8 +912,8 @@
werrstr(""); /* clear spurious error messages */
}
-Rune *
-selrange(Rune *r, int x0, int y0, int x1, int y1)
+char*
+selrange(char *d, int x0, int y0, int x1, int y1)
{
Rune *s, *e;
int z, p;
@@ -925,28 +925,25 @@
if(*s == '\n')
z = p = 0;
else if(p++ == 0){
- while(z-- > 0) *r++ = ' ';
+ while(z-- > 0) *d++ = ' ';
}
- *r++ = *s;
+ d += runetochar(d, s);
} else {
z++;
}
}
- *r = 0;
- return r;
+ return d;
}
-Rune*
-selrunes(void)
+char*
+selection(void)
{
- Rune *r, *p;
- int sz;
+ char *s, *p;
int y;
/* generous, but we can spare a few bytes for a few microseconds */
- sz = xmax*(selrect.max.y - selrect.min.y + 2) + 1;
- r = p = malloc(sizeof(Rune)*sz + 1);
- if(!r)
+ s = p = malloc(UTFmax*(xmax+1)*(Dy(selrect)+1)+1);
+ if(s == nil)
return nil;
if(blocksel){
for(y = selrect.min.y; y <= selrect.max.y; y++){
@@ -953,11 +950,11 @@
p = selrange(p, selrect.min.x, y, selrect.max.x, y);
*p++ = '\n';
}
- *p = 0;
+ } else {
+ p = selrange(p, selrect.min.x, selrect.min.y, selrect.max.x, selrect.max.y);
}
- else
- selrange(r, selrect.min.x, selrect.min.y, selrect.max.x, selrect.max.y);
- return r;
+ *p = 0;
+ return s;
}
void
@@ -964,17 +961,17 @@
snarfsel(void)
{
Biobuf *b;
- Rune *r;
+ char *s;
- if((r = selrunes()) == nil)
+ if((s = selection()) == nil)
return;
if((b = Bopen("/dev/snarf", OWRITE|OTRUNC)) == nil){
- free(r);
+ free(s);
return;
}
- Bprint(b, "%S", r);
+ Bprint(b, "%s", s);
Bterm(b);
- free(r);
+ free(s);
}
void
@@ -981,16 +978,10 @@
plumbsel(void)
{
char *s, wdir[1024];
- Rune *r;
int plumb;
- if((r = selrunes()) == nil)
+ if((s = selection()) == nil)
return;
- if((s = smprint("%S", r)) == nil){
- free(r);
- return;
- }
- free(r);
if(getwd(wdir, sizeof wdir) == nil){
free(s);
return;
@@ -1022,29 +1013,42 @@
}
void
-selection(void)
+select(Point p, Point q)
{
- Point p, q;
- int y;
+ if(onscreenr(p.x, p.y) > onscreenr(q.x, q.y)){
+ select(q, p);
+ return;
+ }
+ unselect();
+ if(p.y < 0 || p.y > ymax)
+ return;
+ if(p.y < 0){
+ p.y = 0;
+ if(!blocksel) p.x = 0;
+ }
+ if(q.y > ymax){
+ q.y = ymax;
+ if(!blocksel) q.x = xmax+1;
+ }
+ if(p.x < 0)
+ p.x = 0;
+ if(q.x > xmax+1)
+ q.x = xmax+1;
+ selrect = Rpt(p, q);
+ for(; p.y <= q.y; p.y++)
+ screenchange(p.y) = 1;
+}
+void
+selecting(void)
+{
+ Point p;
p = pos(mc->xy);
do{
- /* Clear the old selection rectangle. */
- unselect();
- q = pos(mc->xy);
- if(onscreenr(p.x, p.y) > onscreenr(q.x, q.y)){
- selrect.min = q;
- selrect.max = p;
- } else {
- selrect.min = p;
- selrect.max = q;
- }
- /* And mark the new one as changed. */
- for(y = selrect.min.y; y <= selrect.max.y; y++)
- screenchange(y) = 1;
- readmouse(mc);
drawscreen();
+ readmouse(mc);
+ select(p, pos(mc->xy));
} while(button1());
switch(mc->buttons & 0x7){
case 3: snarfsel(); break;
@@ -1255,18 +1259,7 @@
/* move selection */
selrect.min.y -= d;
selrect.max.y -= d;
- if(selrect.max.y < 0 || selrect.min.y > ymax)
- selrect = ZR;
- else {
- if(selrect.min.y < 0){
- selrect.min.y = 0;
- if(!blocksel) selrect.min.x = 0;
- }
- if(selrect.max.y > ymax){
- selrect.max.y = ymax;
- if(!blocksel) selrect.max.x = xmax+1;
- }
- }
+ select(selrect.min, selrect.max);
clear(0, cy, xmax+1, cy+1);
}