terminal: cursor position setting, erase screen commands

This commit is contained in:
Vadim Lopatin 2016-05-30 15:53:29 +03:00
parent c4869ca3c0
commit 7055f48cc5
2 changed files with 238 additions and 34 deletions

View File

@ -161,11 +161,23 @@ class OutputPanel : DockWindow {
_tabs.addTab(_terminalWidget, "Output"d);
_terminalWidget.write("Hello\nSecond line\nTest\n"d);
_terminalWidget.write("SomeString 123456789\rAwesomeString\n"d); // test \r
// testing tabs
_terminalWidget.write("id\tname\tdescription\n"d);
_terminalWidget.write("1\tFoo\tFoo line\n"d);
_terminalWidget.write("2\tBar\tBar line\n"d);
_terminalWidget.write("3\tFoobar\tFoo bar line\n"d);
_terminalWidget.write("Testing very long line. Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.\n"d);
_terminalWidget.write("\n\n\n"d);
// testing line wrapping
_terminalWidget.write("Testing very long line. Юникод. Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.\n"c);
// testing cursor position changes
_terminalWidget.write("\x1b[4;4HCURSOR(4,4)\x1b[HHOME\x1b[B*A\x1b[B*B\x1b[5C\x1b[D***\x1b[A*UP\x1b[3B*DOWN"d);
//_terminalWidget.write("\x1b[Jerased down"d);
//_terminalWidget.write("\x1b[1Jerased up"d);
//_terminalWidget.write("\x1b[2Jerased screen"d);
//_terminalWidget.write("\x1b[Kerased eol"d);
//_terminalWidget.write("\x1b[1Kerased bol"d);
//_terminalWidget.write("\x1b[2Kerased line"d);
//_terminalWidget.write("Юникод Unicode"d);
return _tabs;
}

View File

@ -78,6 +78,73 @@ struct TerminalContent {
int cursorx;
int cursory;
bool focused;
bool _lineWrap = true;
@property void lineWrap(bool v) {
_lineWrap = v;
}
void resetTerminal() {
for (int i = topLine; i < cast(int)lines.length; i++) {
lines[i] = TerminalLine.init;
}
cursorx = 0;
cursory = topLine;
}
@property int screenTopLine() {
int y = cast(int)lines.length - height;
if (y < 0)
y = 0;
return y;
}
void eraseScreen(int direction, bool forLine) {
if (forLine) {
for (int x = 0; x < width; x++) {
if ((direction == 1 && x <= cursorx) || (direction < 1 && x >= cursorx) || (direction == 2))
putCharAt(' ', x, cursory);
}
} else {
int screenTop = screenTopLine;
for (int y = 0; y < height; y++) {
int yy = screenTop + y;
if ((direction == 1 && yy <= cursory) || (direction < 1 && yy >= cursory) || (direction == 2)) {
for (int x = 0; x < width; x++) {
putCharAt(' ', x, yy);
}
}
}
if (direction == 2) {
cursorx = 0;
cursory = screenTop;
}
}
}
void moveCursorBy(int x, int y) {
if (x) {
cursorx += x;
if (cursorx < 0)
cursorx = 0;
if (cursorx > width)
cursorx = width;
} else if (y) {
int screenTop = screenTopLine;
cursory += y;
if (cursory < screenTop)
cursory = screenTop;
else if (cursory >= screenTop + height)
cursory = screenTop + height - 1;
}
}
void moveCursorTo(int x, int y) {
int screenTop = screenTopLine;
if (x < 0 || y < 0) {
cursorx = 0;
cursory = screenTop;
return;
}
if (x >= 1 && x <= width + 1 && y >= 1 && x <= height) {
cursorx = x - 1;
cursory = screenTop + y - 1;
}
}
void layout(FontRef font, Rect rc) {
this.rc = rc;
this.font = font;
@ -108,7 +175,7 @@ struct TerminalContent {
TerminalLine * p = &lines[i + topLine];
// draw line in rect
for (int x = 0; x < width; x++) {
bool isCursorPos = x == cursorx && i + topLine == cursory + screenTopLine;
bool isCursorPos = x == cursorx && i + topLine == cursory;
TerminalChar ch = x < p.line.length ? p.line[x] : TerminalChar.init;
uint bgcolor = attrToColor(ch.attr.bgColor);
uint textcolor = attrToColor(ch.attr.textColor);
@ -134,35 +201,30 @@ struct TerminalContent {
lineRect.top = lineRect.bottom;
}
}
TerminalLine * getLine(ref int yy) {
if (yy < 0)
yy = 0;
if (yy > height)
yy = height;
if (yy == height) {
topLine++;
yy--;
}
int y = yy;
y = topLine + y;
void clearExtraLines(ref int yy) {
int y = cast(int)lines.length;
if (y >= maxBufferLines) {
int delta = y - maxBufferLines;
for (uint i = 0; i + delta < maxBufferLines && i + delta < lines.length; i++) {
lines[i] = lines[i + delta];
}
y -= delta;
if (lines.length < maxBufferLines) {
size_t oldlen = lines.length;
lines.length = maxBufferLines;
for(auto i = oldlen; i < lines.length; i++)
lines[i] = TerminalLine.init;
}
lines.length = lines.length - delta;
yy -= delta;
topLine -= delta;
if (topLine < 0)
topLine = 0;
}
if (cast(uint)y >= lines.length) {
for (auto i = lines.length; i <= y; i++)
lines ~= TerminalLine.init;
}
TerminalLine * getLine(ref int yy) {
if (yy < 0)
yy = 0;
while(yy >= cast(int)lines.length) {
lines ~= TerminalLine.init;
}
return &lines[y];
clearExtraLines(yy);
return &lines[yy];
}
void putCharAt(dchar ch, ref int x, ref int y) {
if (x < 0)
@ -175,20 +237,36 @@ struct TerminalContent {
x = 0;
}
line.putCharAt(ch, x, currentAttr);
ensureCursorIsVisible();
}
int tabSize = 8;
// supports printed characters and \r \n \t
void putChar(dchar ch) {
if (ch == '\r') {
cursorx = 0;
if (ch == '\a') {
// bell
return;
}
if (ch == '\n') {
if (ch == '\b') {
// backspace
if (cursorx > 0) {
cursorx--;
putCharAt(' ', cursorx, cursory);
ensureCursorIsVisible();
}
return;
}
if (ch == '\r') {
cursorx = 0;
ensureCursorIsVisible();
return;
}
if (ch == '\n' || ch == '\f' || ch == '\v') {
TerminalLine * line = getLine(cursory);
line.markLineEol();
cursory++;
line = getLine(cursory);
cursorx = 0;
ensureCursorIsVisible();
return;
}
if (ch == '\t') {
@ -205,15 +283,25 @@ struct TerminalContent {
cursorx++;
}
}
ensureCursorIsVisible();
return;
}
putCharAt(ch, cursorx, cursory);
cursorx++;
ensureCursorIsVisible();
}
void ensureCursorIsVisible() {
topLine = cast(int)lines.length - height;
if (topLine < 0)
topLine = 0;
if (cursory < topLine)
cursory = topLine;
}
void updateScrollBar(ScrollBar sb) {
sb.pageSize = height;
sb.maxValue = lines.length ? lines.length - 1 : 0;
sb.maxValue = cast(int)lines.length;
sb.position = topLine;
}
@ -345,7 +433,7 @@ class TerminalWidget : WidgetGroup, OnScrollHandler {
dchar[] decoded;
decoded.assumeSafeAppend;
dchar ch = 0;
for (;;) {
while (index < outputBuffer.length) {
size_t oldindex = index;
try {
ch = decode(outputBuffer, index);
@ -370,6 +458,19 @@ class TerminalWidget : WidgetGroup, OnScrollHandler {
write(cast(dstring)decoded);
}
static bool parseParam(dchar[] buf, ref int index, ref int value) {
if (index >= buf.length)
return false;
if (buf[index] < '0' || buf[index] > '9')
return false;
value = 0;
while (index < buf.length && buf[index] >= '0' && buf[index] <= '9') {
value = value * 10 + (buf[index] - '0');
index++;
}
return true;
}
private dchar[] outputChars;
// write utf32
void write(dstring chars) {
@ -381,13 +482,104 @@ class TerminalWidget : WidgetGroup, OnScrollHandler {
return;
uint i = 0;
for (; i < outputChars.length; i++) {
int ch = outputChars[i];
bool unfinished = false;
dchar ch = outputChars[i];
dchar ch2 = i + 1 < outputChars.length ? outputChars[i + 1] : 0;
dchar ch3 = i + 2 < outputChars.length ? outputChars[i + 2] : 0;
//dchar ch4 = i + 3 < outputChars.length ? outputChars[i + 3] : 0;
if (ch < ' ') {
// control character
switch(ch) {
case '\r':
case '\n':
case '\t':
if (ch == 27) {
if (ch2 == 0)
break; // unfinished ESC sequence
// ESC sequence
if (ch2 == '[') {
// ESC [
if (!ch3)
break; // unfinished
int param1 = -1;
int param2 = -1;
int index = i + 2;
bool questionMark = false;
if (index < outputChars.length && outputChars[index] == '?') {
questionMark = true;
index++;
}
parseParam(outputChars, index, param1);
if (index < outputChars.length && outputChars[index] == ';') {
index++;
parseParam(outputChars, index, param2);
}
if (index >= outputChars.length)
break; // unfinished sequence: not enough chars
int param1def1 = param1 >= 1 ? param1 : 1;
ch3 = outputChars[index];
i = index;
// command is parsed completely, ch3 == command type char
// ESC[7h and ESC[7l -- enable/disable line wrap
if (param1 == '7' && (ch3 == 'h' || ch3 == 'l')) {
_content.lineWrap(ch3 == 'h');
continue;
}
if (ch3 == 'H' || ch3 == 'f') {
_content.moveCursorTo(param2, param1);
continue;
}
if (ch3 == 'A') { // cursor up
_content.moveCursorBy(0, -param1def1);
continue;
}
if (ch3 == 'B') { // cursor down
_content.moveCursorBy(0, param1def1);
continue;
}
if (ch3 == 'C') { // cursor forward
_content.moveCursorBy(param1def1, 0);
continue;
}
if (ch3 == 'D') { // cursor back
_content.moveCursorBy(-param1def1, 0);
continue;
}
if (ch3 == 'K' || ch3 == 'J') {
_content.eraseScreen(param1, ch3 == 'K');
continue;
}
} else switch(ch2) {
case 'c':
_content.resetTerminal();
i++;
break;
case '=': // Set alternate keypad mode
case '>': // Set numeric keypad mode
case 'N': // Set single shift 2
case 'O': // Set single shift 3
case 'H': // Set a tab at the current column
case '<': // Enter/exit ANSI mode (VT52)
i++;
// ignore
break;
case '(': // default font
case ')': // alternate font
i++;
i++;
// ignore
break;
default:
// unsupported
break;
}
if (unfinished)
break;
} else switch(ch) {
case '\a': // bell
case '\f': // form feed
case '\v': // vtab
case '\r': // cr
case '\n': // lf
case '\t': // tab
case '\b': // backspace
_content.putChar(ch);
break;
default: