mirror of https://github.com/adamdruppe/arsd.git
make update cursor position public
This commit is contained in:
parent
ef4ed76960
commit
da6563a5de
289
terminal.d
289
terminal.d
|
@ -1881,12 +1881,17 @@ http://msdn.microsoft.com/en-us/library/windows/desktop/ms683193%28v=vs.85%29.as
|
||||||
|
|
||||||
// FIXME: add moveRelative
|
// FIXME: add moveRelative
|
||||||
|
|
||||||
/// The current x position of the output cursor. 0 == leftmost column
|
/++
|
||||||
|
The current cached x and y positions of the output cursor. 0 == leftmost column for x and topmost row for y.
|
||||||
|
|
||||||
|
Please note that the cached position is not necessarily accurate. You may consider calling [updateCursorPosition]
|
||||||
|
first to ask the terminal for its authoritative answer.
|
||||||
|
+/
|
||||||
@property int cursorX() {
|
@property int cursorX() {
|
||||||
return _cursorX;
|
return _cursorX;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// The current y position of the output cursor. 0 == topmost row
|
/// ditto
|
||||||
@property int cursorY() {
|
@property int cursorY() {
|
||||||
return _cursorY;
|
return _cursorY;
|
||||||
}
|
}
|
||||||
|
@ -2396,6 +2401,149 @@ http://msdn.microsoft.com/en-us/library/windows/desktop/ms683193%28v=vs.85%29.as
|
||||||
return getline(prompt, echoChar, prefilledData);
|
return getline(prompt, echoChar, prefilledData);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/++
|
||||||
|
Forces [cursorX] and [cursorY] to resync from the terminal.
|
||||||
|
|
||||||
|
History:
|
||||||
|
Added January 8, 2023
|
||||||
|
+/
|
||||||
|
void updateCursorPosition() {
|
||||||
|
auto terminal = &this;
|
||||||
|
|
||||||
|
terminal.flush();
|
||||||
|
|
||||||
|
// then get the current cursor position to start fresh
|
||||||
|
version(TerminalDirectToEmulator) {
|
||||||
|
if(!terminal.usingDirectEmulator)
|
||||||
|
return updateCursorPosition_impl();
|
||||||
|
|
||||||
|
if(terminal.pipeThroughStdOut) {
|
||||||
|
terminal.tew.terminalEmulator.waitingForInboundSync = true;
|
||||||
|
terminal.writeStringRaw("\xff");
|
||||||
|
terminal.flush();
|
||||||
|
if(windowGone) forceTermination();
|
||||||
|
terminal.tew.terminalEmulator.syncSignal.wait();
|
||||||
|
}
|
||||||
|
|
||||||
|
terminal._cursorX = terminal.tew.terminalEmulator.cursorX;
|
||||||
|
terminal._cursorY = terminal.tew.terminalEmulator.cursorY;
|
||||||
|
} else
|
||||||
|
updateCursorPosition_impl();
|
||||||
|
}
|
||||||
|
private void updateCursorPosition_impl() {
|
||||||
|
auto terminal = &this;
|
||||||
|
version(Win32Console) {
|
||||||
|
CONSOLE_SCREEN_BUFFER_INFO info;
|
||||||
|
GetConsoleScreenBufferInfo(terminal.hConsole, &info);
|
||||||
|
_cursorX = info.dwCursorPosition.X;
|
||||||
|
_cursorY = info.dwCursorPosition.Y;
|
||||||
|
} else version(Posix) {
|
||||||
|
// request current cursor position
|
||||||
|
|
||||||
|
// we have to turn off cooked mode to get this answer, otherwise it will all
|
||||||
|
// be messed up. (I hate unix terminals, the Windows way is so much easer.)
|
||||||
|
|
||||||
|
// We also can't use RealTimeConsoleInput here because it also does event loop stuff
|
||||||
|
// which would be broken by the child destructor :( (maybe that should be a FIXME)
|
||||||
|
|
||||||
|
/+
|
||||||
|
if(rtci !is null) {
|
||||||
|
while(rtci.timedCheckForInput_bypassingBuffer(1000))
|
||||||
|
rtci.inputQueue ~= rtci.readNextEvents();
|
||||||
|
}
|
||||||
|
+/
|
||||||
|
|
||||||
|
ubyte[128] hack2;
|
||||||
|
termios old;
|
||||||
|
ubyte[128] hack;
|
||||||
|
tcgetattr(terminal.fdIn, &old);
|
||||||
|
auto n = old;
|
||||||
|
n.c_lflag &= ~(ICANON | ECHO);
|
||||||
|
tcsetattr(terminal.fdIn, TCSANOW, &n);
|
||||||
|
scope(exit)
|
||||||
|
tcsetattr(terminal.fdIn, TCSANOW, &old);
|
||||||
|
|
||||||
|
|
||||||
|
terminal.writeStringRaw("\033[6n");
|
||||||
|
terminal.flush();
|
||||||
|
|
||||||
|
import std.conv;
|
||||||
|
import core.stdc.errno;
|
||||||
|
|
||||||
|
import core.sys.posix.unistd;
|
||||||
|
|
||||||
|
ubyte readOne() {
|
||||||
|
ubyte[1] buffer;
|
||||||
|
int tries = 0;
|
||||||
|
try_again:
|
||||||
|
if(tries > 30)
|
||||||
|
throw new Exception("terminal reply timed out");
|
||||||
|
auto len = read(terminal.fdIn, buffer.ptr, buffer.length);
|
||||||
|
if(len == -1) {
|
||||||
|
if(errno == EINTR)
|
||||||
|
goto try_again;
|
||||||
|
if(errno == EAGAIN || errno == EWOULDBLOCK) {
|
||||||
|
import core.thread;
|
||||||
|
Thread.sleep(10.msecs);
|
||||||
|
tries++;
|
||||||
|
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));
|
||||||
|
}
|
||||||
|
|
||||||
|
return buffer[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
nextEscape:
|
||||||
|
while(readOne() != '\033') {}
|
||||||
|
if(readOne() != '[')
|
||||||
|
goto nextEscape;
|
||||||
|
|
||||||
|
int x, y;
|
||||||
|
|
||||||
|
// 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;
|
||||||
|
|
||||||
|
b = readOne();
|
||||||
|
nextNumberX:
|
||||||
|
if(b >= '0' && b <= '9') {
|
||||||
|
x *= 10;
|
||||||
|
x += b - '0';
|
||||||
|
} else goto nextEscape;
|
||||||
|
|
||||||
|
b = readOne();
|
||||||
|
// another digit
|
||||||
|
if(b >= '0' && b <= '9')
|
||||||
|
goto nextNumberX;
|
||||||
|
|
||||||
|
if(b != 'R')
|
||||||
|
goto nextEscape; // it wasn't the right thing it after all
|
||||||
|
|
||||||
|
_cursorX = x - 1;
|
||||||
|
_cursorY = y - 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/++
|
/++
|
||||||
|
@ -6211,141 +6359,10 @@ class LineGetter {
|
||||||
}
|
}
|
||||||
|
|
||||||
protected void updateCursorPosition() {
|
protected void updateCursorPosition() {
|
||||||
terminal.flush();
|
terminal.updateCursorPosition();
|
||||||
|
|
||||||
// then get the current cursor position to start fresh
|
startOfLineX = terminal.cursorX;
|
||||||
version(TerminalDirectToEmulator) {
|
startOfLineY = terminal.cursorY;
|
||||||
if(!terminal.usingDirectEmulator)
|
|
||||||
return updateCursorPosition_impl();
|
|
||||||
|
|
||||||
if(terminal.pipeThroughStdOut) {
|
|
||||||
terminal.tew.terminalEmulator.waitingForInboundSync = true;
|
|
||||||
terminal.writeStringRaw("\xff");
|
|
||||||
terminal.flush();
|
|
||||||
if(windowGone) forceTermination();
|
|
||||||
terminal.tew.terminalEmulator.syncSignal.wait();
|
|
||||||
}
|
|
||||||
|
|
||||||
startOfLineX = terminal.tew.terminalEmulator.cursorX;
|
|
||||||
startOfLineY = terminal.tew.terminalEmulator.cursorY;
|
|
||||||
} else
|
|
||||||
updateCursorPosition_impl();
|
|
||||||
}
|
|
||||||
private void updateCursorPosition_impl() {
|
|
||||||
version(Win32Console) {
|
|
||||||
CONSOLE_SCREEN_BUFFER_INFO info;
|
|
||||||
GetConsoleScreenBufferInfo(terminal.hConsole, &info);
|
|
||||||
startOfLineX = info.dwCursorPosition.X;
|
|
||||||
startOfLineY = info.dwCursorPosition.Y;
|
|
||||||
} else version(Posix) {
|
|
||||||
// request current cursor position
|
|
||||||
|
|
||||||
// we have to turn off cooked mode to get this answer, otherwise it will all
|
|
||||||
// be messed up. (I hate unix terminals, the Windows way is so much easer.)
|
|
||||||
|
|
||||||
// We also can't use RealTimeConsoleInput here because it also does event loop stuff
|
|
||||||
// which would be broken by the child destructor :( (maybe that should be a FIXME)
|
|
||||||
|
|
||||||
/+
|
|
||||||
if(rtci !is null) {
|
|
||||||
while(rtci.timedCheckForInput_bypassingBuffer(1000))
|
|
||||||
rtci.inputQueue ~= rtci.readNextEvents();
|
|
||||||
}
|
|
||||||
+/
|
|
||||||
|
|
||||||
ubyte[128] hack2;
|
|
||||||
termios old;
|
|
||||||
ubyte[128] hack;
|
|
||||||
tcgetattr(terminal.fdIn, &old);
|
|
||||||
auto n = old;
|
|
||||||
n.c_lflag &= ~(ICANON | ECHO);
|
|
||||||
tcsetattr(terminal.fdIn, TCSANOW, &n);
|
|
||||||
scope(exit)
|
|
||||||
tcsetattr(terminal.fdIn, TCSANOW, &old);
|
|
||||||
|
|
||||||
|
|
||||||
terminal.writeStringRaw("\033[6n");
|
|
||||||
terminal.flush();
|
|
||||||
|
|
||||||
import std.conv;
|
|
||||||
import core.stdc.errno;
|
|
||||||
|
|
||||||
import core.sys.posix.unistd;
|
|
||||||
|
|
||||||
ubyte readOne() {
|
|
||||||
ubyte[1] buffer;
|
|
||||||
int tries = 0;
|
|
||||||
try_again:
|
|
||||||
if(tries > 30)
|
|
||||||
throw new Exception("terminal reply timed out");
|
|
||||||
auto len = read(terminal.fdIn, buffer.ptr, buffer.length);
|
|
||||||
if(len == -1) {
|
|
||||||
if(errno == EINTR)
|
|
||||||
goto try_again;
|
|
||||||
if(errno == EAGAIN || errno == EWOULDBLOCK) {
|
|
||||||
import core.thread;
|
|
||||||
Thread.sleep(10.msecs);
|
|
||||||
tries++;
|
|
||||||
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));
|
|
||||||
}
|
|
||||||
|
|
||||||
return buffer[0];
|
|
||||||
}
|
|
||||||
|
|
||||||
nextEscape:
|
|
||||||
while(readOne() != '\033') {}
|
|
||||||
if(readOne() != '[')
|
|
||||||
goto nextEscape;
|
|
||||||
|
|
||||||
int x, y;
|
|
||||||
|
|
||||||
// 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;
|
|
||||||
|
|
||||||
b = readOne();
|
|
||||||
nextNumberX:
|
|
||||||
if(b >= '0' && b <= '9') {
|
|
||||||
x *= 10;
|
|
||||||
x += b - '0';
|
|
||||||
} else goto nextEscape;
|
|
||||||
|
|
||||||
b = readOne();
|
|
||||||
// another digit
|
|
||||||
if(b >= '0' && b <= '9')
|
|
||||||
goto nextNumberX;
|
|
||||||
|
|
||||||
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
|
|
||||||
terminal._cursorX = startOfLineX;
|
|
||||||
terminal._cursorY = startOfLineY;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Text killed with C-w/C-u/C-k/C-backspace, to be restored by C-y
|
// Text killed with C-w/C-u/C-k/C-backspace, to be restored by C-y
|
||||||
|
|
Loading…
Reference in New Issue