mirror of https://github.com/adamdruppe/arsd.git
more robust cursor position query
This commit is contained in:
parent
a90f344477
commit
2ed6b50aa6
86
terminal.d
86
terminal.d
|
@ -3582,6 +3582,7 @@ void main() {
|
||||||
getter.prompt = "> ";
|
getter.prompt = "> ";
|
||||||
getter.history = ["abcdefghijklmnopqrstuvwzyz1234567890ABCDEFGHIJKLMNOPQRSTUVWXYZ"];
|
getter.history = ["abcdefghijklmnopqrstuvwzyz1234567890ABCDEFGHIJKLMNOPQRSTUVWXYZ"];
|
||||||
terminal.writeln("\n" ~ getter.getline());
|
terminal.writeln("\n" ~ getter.getline());
|
||||||
|
|
||||||
terminal.writeln("\n" ~ getter.getline());
|
terminal.writeln("\n" ~ getter.getline());
|
||||||
terminal.writeln("\n" ~ getter.getline());
|
terminal.writeln("\n" ~ getter.getline());
|
||||||
getter.dispose();
|
getter.dispose();
|
||||||
|
@ -4609,49 +4610,72 @@ class LineGetter {
|
||||||
import core.stdc.errno;
|
import core.stdc.errno;
|
||||||
|
|
||||||
import core.sys.posix.unistd;
|
import core.sys.posix.unistd;
|
||||||
// reading directly to bypass any buffering
|
|
||||||
int retries = 16;
|
ubyte readOne() {
|
||||||
ubyte[16] buffer;
|
ubyte[1] buffer;
|
||||||
try_again:
|
int tries = 0;
|
||||||
auto len = read(terminal.fdIn, buffer.ptr, buffer.length);
|
try_again:
|
||||||
if(len <= 0) {
|
if(tries > 30)
|
||||||
|
throw new Exception("terminal reply timed out");
|
||||||
|
auto len = read(terminal.fdIn, buffer.ptr, buffer.length);
|
||||||
if(len == -1) {
|
if(len == -1) {
|
||||||
if(errno == EINTR)
|
if(errno == EINTR)
|
||||||
goto try_again;
|
goto try_again;
|
||||||
if(errno == EAGAIN || errno == EWOULDBLOCK) {
|
if(errno == EAGAIN || errno == EWOULDBLOCK) {
|
||||||
import core.thread;
|
import core.thread;
|
||||||
Thread.sleep(10.msecs);
|
Thread.sleep(10.msecs);
|
||||||
|
tries++;
|
||||||
goto try_again;
|
goto try_again;
|
||||||
}
|
}
|
||||||
|
} else if(len == 0) {
|
||||||
|
throw new Exception("Couldn't get cursor position to initialize get line " ~ to!string(len) ~ " " ~ to!string(errno));
|
||||||
}
|
}
|
||||||
throw new Exception("Couldn't get cursor position to initialize get line " ~ to!string(len) ~ " " ~ to!string(errno));
|
|
||||||
}
|
|
||||||
regot:
|
|
||||||
auto got = buffer[0 .. len];
|
|
||||||
if(got.length < 6) {
|
|
||||||
auto len2 = read(terminal.fdIn, &buffer[len], buffer.length - len);
|
|
||||||
if(len2 <= 0)
|
|
||||||
throw new Exception("not enough cursor reply answer");
|
|
||||||
else {
|
|
||||||
len += len2;
|
|
||||||
goto regot;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if(got[0] != '\033' || got[1] != '[' || got[$-1] != 'R') {
|
|
||||||
retries--;
|
|
||||||
if(retries > 0)
|
|
||||||
goto try_again;
|
|
||||||
throw new Exception("wrong answer for cursor position " ~ cast(string) got[1 .. $]);
|
|
||||||
}
|
|
||||||
auto gots = cast(char[]) got[2 .. $-1];
|
|
||||||
|
|
||||||
import std.string;
|
return buffer[0];
|
||||||
|
}
|
||||||
|
|
||||||
auto pieces = split(gots, ";");
|
nextEscape:
|
||||||
if(pieces.length != 2) throw new Exception("wtf wrong answer on cursor position");
|
while(readOne() != '\033') {}
|
||||||
|
if(readOne() != '[')
|
||||||
|
goto nextEscape;
|
||||||
|
|
||||||
startOfLineX = to!int(pieces[1]) - 1;
|
int x, y;
|
||||||
startOfLineY = to!int(pieces[0]) - 1;
|
|
||||||
|
// now we should have some numbers being like yyy;xxxR
|
||||||
|
// but there may be a ? in there too; DEC private mode format
|
||||||
|
// of the very same data.
|
||||||
|
|
||||||
|
x = 0;
|
||||||
|
y = 0;
|
||||||
|
|
||||||
|
auto b = readOne();
|
||||||
|
|
||||||
|
if(b == '?')
|
||||||
|
b = readOne(); // no big deal, just ignore and continue
|
||||||
|
|
||||||
|
nextNumberY:
|
||||||
|
if(b >= '0' || b <= '9') {
|
||||||
|
y *= 10;
|
||||||
|
y += b - '0';
|
||||||
|
} else goto nextEscape;
|
||||||
|
|
||||||
|
b = readOne();
|
||||||
|
if(b != ';')
|
||||||
|
goto nextNumberY;
|
||||||
|
|
||||||
|
nextNumberX:
|
||||||
|
b = readOne();
|
||||||
|
if(b >= '0' || b <= '9') {
|
||||||
|
x *= 10;
|
||||||
|
x += b - '0';
|
||||||
|
} else goto nextEscape;
|
||||||
|
|
||||||
|
b = readOne();
|
||||||
|
if(b != 'R')
|
||||||
|
goto nextEscape; // it wasn't the right thing it after all
|
||||||
|
|
||||||
|
startOfLineX = x - 1;
|
||||||
|
startOfLineY = y - 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// updating these too because I can with the more accurate info from above
|
// updating these too because I can with the more accurate info from above
|
||||||
|
|
Loading…
Reference in New Issue