mirror of https://github.com/adamdruppe/arsd.git
slightly better multi line editor scrolling
This commit is contained in:
parent
c2c50fd576
commit
a7209664e7
154
terminal.d
154
terminal.d
|
@ -5386,12 +5386,17 @@ class LineGetter {
|
||||||
private dchar[] line;
|
private dchar[] line;
|
||||||
private int cursorPosition = 0;
|
private int cursorPosition = 0;
|
||||||
private int horizontalScrollPosition = 0;
|
private int horizontalScrollPosition = 0;
|
||||||
|
private int verticalScrollPosition = 0;
|
||||||
|
|
||||||
private void scrollToEnd() {
|
private void scrollToEnd() {
|
||||||
horizontalScrollPosition = (cast(int) line.length);
|
if(multiLineMode) {
|
||||||
horizontalScrollPosition -= availableLineLength();
|
// FIXME
|
||||||
if(horizontalScrollPosition < 0)
|
} else {
|
||||||
horizontalScrollPosition = 0;
|
horizontalScrollPosition = (cast(int) line.length);
|
||||||
|
horizontalScrollPosition -= availableLineLength();
|
||||||
|
if(horizontalScrollPosition < 0)
|
||||||
|
horizontalScrollPosition = 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// used for redrawing the line in the right place
|
// used for redrawing the line in the right place
|
||||||
|
@ -5445,8 +5450,12 @@ class LineGetter {
|
||||||
}
|
}
|
||||||
cursorPosition++;
|
cursorPosition++;
|
||||||
|
|
||||||
if(cursorPosition > horizontalScrollPosition + availableLineLength())
|
if(multiLineMode) {
|
||||||
horizontalScrollPosition++;
|
// FIXME
|
||||||
|
} else {
|
||||||
|
if(cursorPosition > horizontalScrollPosition + availableLineLength())
|
||||||
|
horizontalScrollPosition++;
|
||||||
|
}
|
||||||
|
|
||||||
lineChanged = true;
|
lineChanged = true;
|
||||||
}
|
}
|
||||||
|
@ -5561,8 +5570,43 @@ class LineGetter {
|
||||||
}
|
}
|
||||||
|
|
||||||
private void maybePositionCursor() {
|
private void maybePositionCursor() {
|
||||||
if(cursorPosition < horizontalScrollPosition || cursorPosition > horizontalScrollPosition + availableLineLength()) {
|
if(multiLineMode) {
|
||||||
positionCursor();
|
// omg this is so bad
|
||||||
|
// and it more accurately sets scroll position
|
||||||
|
int x, y;
|
||||||
|
foreach(idx, ch; line) {
|
||||||
|
if(idx == cursorPosition)
|
||||||
|
break;
|
||||||
|
if(ch == '\n') {
|
||||||
|
x = 0;
|
||||||
|
y++;
|
||||||
|
} else {
|
||||||
|
x++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
while(x - horizontalScrollPosition < 0) {
|
||||||
|
horizontalScrollPosition -= terminal.width / 2;
|
||||||
|
if(horizontalScrollPosition < 0)
|
||||||
|
horizontalScrollPosition = 0;
|
||||||
|
}
|
||||||
|
while(y - verticalScrollPosition < 0) {
|
||||||
|
verticalScrollPosition --;
|
||||||
|
if(verticalScrollPosition < 0)
|
||||||
|
verticalScrollPosition = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
while((x - horizontalScrollPosition) >= terminal.width) {
|
||||||
|
horizontalScrollPosition += terminal.width / 2;
|
||||||
|
}
|
||||||
|
while((y - verticalScrollPosition) + 2 >= terminal.height) {
|
||||||
|
verticalScrollPosition ++;
|
||||||
|
}
|
||||||
|
|
||||||
|
} else {
|
||||||
|
if(cursorPosition < horizontalScrollPosition || cursorPosition > horizontalScrollPosition + availableLineLength()) {
|
||||||
|
positionCursor();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -5606,11 +5650,14 @@ class LineGetter {
|
||||||
|
|
||||||
this(LineGetter lg) {
|
this(LineGetter lg) {
|
||||||
this.lg = lg;
|
this.lg = lg;
|
||||||
|
linesRemaining = lg.terminal.height - 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
int written;
|
int written;
|
||||||
int lineLength;
|
int lineLength;
|
||||||
|
|
||||||
|
int linesRemaining;
|
||||||
|
|
||||||
|
|
||||||
Color currentFg_ = Color.DEFAULT;
|
Color currentFg_ = Color.DEFAULT;
|
||||||
Color currentBg_ = Color.DEFAULT;
|
Color currentBg_ = Color.DEFAULT;
|
||||||
|
@ -5658,6 +5705,7 @@ class LineGetter {
|
||||||
if(lg.multiLineMode) {
|
if(lg.multiLineMode) {
|
||||||
if(ch == '\n') {
|
if(ch == '\n') {
|
||||||
lineLength = lg.terminal.width;
|
lineLength = lg.terminal.width;
|
||||||
|
linesRemaining--;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -5673,10 +5721,13 @@ class LineGetter {
|
||||||
}
|
}
|
||||||
|
|
||||||
foreach(idx, dchar ch; towrite) {
|
foreach(idx, dchar ch; towrite) {
|
||||||
|
if(linesRemaining <= 0)
|
||||||
|
break;
|
||||||
if(lineLength <= 0) {
|
if(lineLength <= 0) {
|
||||||
if(lg.multiLineMode) {
|
if(lg.multiLineMode) {
|
||||||
if(ch == '\n')
|
if(ch == '\n') {
|
||||||
lineLength = lg.terminal.width;
|
lineLength = lg.terminal.width;
|
||||||
|
}
|
||||||
continue;
|
continue;
|
||||||
} else
|
} else
|
||||||
break;
|
break;
|
||||||
|
@ -5841,10 +5892,26 @@ class LineGetter {
|
||||||
terminal.color(regularForeground, background);
|
terminal.color(regularForeground, background);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto towrite = line[horizontalScrollPosition .. $];
|
dchar[] towrite;
|
||||||
|
|
||||||
if(multiLineMode) {
|
if(multiLineMode) {
|
||||||
towrite = line[];
|
towrite = line[];
|
||||||
|
if(verticalScrollPosition) {
|
||||||
|
int remaining = verticalScrollPosition;
|
||||||
|
while(towrite.length) {
|
||||||
|
if(towrite[0] == '\n') {
|
||||||
|
towrite = towrite[1 .. $];
|
||||||
|
remaining--;
|
||||||
|
if(remaining == 0)
|
||||||
|
break;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
towrite = towrite[1 .. $];
|
||||||
|
}
|
||||||
|
}
|
||||||
horizontalScrollPosition = 0; // FIXME
|
horizontalScrollPosition = 0; // FIXME
|
||||||
|
} else {
|
||||||
|
towrite = line[horizontalScrollPosition .. $];
|
||||||
}
|
}
|
||||||
auto cursorPositionToDrawX = cursorPosition - horizontalScrollPosition;
|
auto cursorPositionToDrawX = cursorPosition - horizontalScrollPosition;
|
||||||
auto cursorPositionToDrawY = 0;
|
auto cursorPositionToDrawY = 0;
|
||||||
|
@ -5898,8 +5965,8 @@ class LineGetter {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
cri.cursorPositionToDrawX = cursorPositionToDrawX;
|
cri.cursorPositionToDrawX = cursorPositionToDrawX - horizontalScrollPosition;
|
||||||
cri.cursorPositionToDrawY = cursorPositionToDrawY;
|
cri.cursorPositionToDrawY = cursorPositionToDrawY - verticalScrollPosition;
|
||||||
} else {
|
} else {
|
||||||
cri.cursorPositionToDrawX = cursorPositionToDrawX;
|
cri.cursorPositionToDrawX = cursorPositionToDrawX;
|
||||||
cri.cursorPositionToDrawY = cursorPositionToDrawY;
|
cri.cursorPositionToDrawY = cursorPositionToDrawY;
|
||||||
|
@ -5917,6 +5984,7 @@ class LineGetter {
|
||||||
if(!maintainBuffer) {
|
if(!maintainBuffer) {
|
||||||
cursorPosition = 0;
|
cursorPosition = 0;
|
||||||
horizontalScrollPosition = 0;
|
horizontalScrollPosition = 0;
|
||||||
|
verticalScrollPosition = 0;
|
||||||
justHitTab = false;
|
justHitTab = false;
|
||||||
currentHistoryViewPosition = 0;
|
currentHistoryViewPosition = 0;
|
||||||
if(line.length) {
|
if(line.length) {
|
||||||
|
@ -5934,18 +6002,24 @@ class LineGetter {
|
||||||
}
|
}
|
||||||
|
|
||||||
private void positionCursor() {
|
private void positionCursor() {
|
||||||
if(cursorPosition == 0)
|
if(cursorPosition == 0) {
|
||||||
horizontalScrollPosition = 0;
|
horizontalScrollPosition = 0;
|
||||||
else if(cursorPosition == line.length)
|
verticalScrollPosition = 0;
|
||||||
|
} else if(cursorPosition == line.length) {
|
||||||
scrollToEnd();
|
scrollToEnd();
|
||||||
else {
|
} else {
|
||||||
// otherwise just try to center it in the screen
|
if(multiLineMode) {
|
||||||
horizontalScrollPosition = cursorPosition;
|
// FIXME
|
||||||
horizontalScrollPosition -= maximumDrawWidth / 2;
|
maybePositionCursor();
|
||||||
// align on a code point boundary
|
} else {
|
||||||
aligned(horizontalScrollPosition, -1);
|
// otherwise just try to center it in the screen
|
||||||
if(horizontalScrollPosition < 0)
|
horizontalScrollPosition = cursorPosition;
|
||||||
horizontalScrollPosition = 0;
|
horizontalScrollPosition -= maximumDrawWidth / 2;
|
||||||
|
// align on a code point boundary
|
||||||
|
aligned(horizontalScrollPosition, -1);
|
||||||
|
if(horizontalScrollPosition < 0)
|
||||||
|
horizontalScrollPosition = 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -6279,11 +6353,15 @@ class LineGetter {
|
||||||
}
|
}
|
||||||
|
|
||||||
void pageBackward() {
|
void pageBackward() {
|
||||||
|
foreach(count; 0 .. terminal.height)
|
||||||
|
lineBackward();
|
||||||
|
maybePositionCursor();
|
||||||
}
|
}
|
||||||
|
|
||||||
void pageForward() {
|
void pageForward() {
|
||||||
|
foreach(count; 0 .. terminal.height)
|
||||||
|
lineForward();
|
||||||
|
maybePositionCursor();
|
||||||
}
|
}
|
||||||
|
|
||||||
/++
|
/++
|
||||||
|
@ -6432,7 +6510,9 @@ class LineGetter {
|
||||||
line = line[0 .. $ - 1];
|
line = line[0 .. $ - 1];
|
||||||
line.assumeSafeAppend();
|
line.assumeSafeAppend();
|
||||||
|
|
||||||
if(!multiLineMode) {
|
if(multiLineMode) {
|
||||||
|
// FIXME
|
||||||
|
} else {
|
||||||
if(horizontalScrollPosition > cursorPosition - 1)
|
if(horizontalScrollPosition > cursorPosition - 1)
|
||||||
horizontalScrollPosition = cursorPosition - 1 - availableLineLength();
|
horizontalScrollPosition = cursorPosition - 1 - availableLineLength();
|
||||||
if(horizontalScrollPosition < 0)
|
if(horizontalScrollPosition < 0)
|
||||||
|
@ -6629,9 +6709,10 @@ class LineGetter {
|
||||||
goto default;
|
goto default;
|
||||||
case KeyboardEvent.Key.UpArrow:
|
case KeyboardEvent.Key.UpArrow:
|
||||||
justHitTab = justKilled = false;
|
justHitTab = justKilled = false;
|
||||||
if(multiLineMode)
|
if(multiLineMode) {
|
||||||
lineBackward();
|
lineBackward();
|
||||||
else
|
maybePositionCursor();
|
||||||
|
} else
|
||||||
loadFromHistory(currentHistoryViewPosition + 1);
|
loadFromHistory(currentHistoryViewPosition + 1);
|
||||||
redraw();
|
redraw();
|
||||||
break;
|
break;
|
||||||
|
@ -6641,9 +6722,10 @@ class LineGetter {
|
||||||
goto default;
|
goto default;
|
||||||
case KeyboardEvent.Key.DownArrow:
|
case KeyboardEvent.Key.DownArrow:
|
||||||
justHitTab = justKilled = false;
|
justHitTab = justKilled = false;
|
||||||
if(multiLineMode)
|
if(multiLineMode) {
|
||||||
lineForward();
|
lineForward();
|
||||||
else
|
maybePositionCursor();
|
||||||
|
} else
|
||||||
loadFromHistory(currentHistoryViewPosition - 1);
|
loadFromHistory(currentHistoryViewPosition - 1);
|
||||||
redraw();
|
redraw();
|
||||||
break;
|
break;
|
||||||
|
@ -6815,7 +6897,9 @@ class LineGetter {
|
||||||
auto me = e.mouseEvent;
|
auto me = e.mouseEvent;
|
||||||
if(me.eventType == MouseEvent.Type.Pressed) {
|
if(me.eventType == MouseEvent.Type.Pressed) {
|
||||||
if(me.buttons & MouseEvent.Button.Left) {
|
if(me.buttons & MouseEvent.Button.Left) {
|
||||||
if(me.y == startOfLineY) {
|
if(multiLineMode) {
|
||||||
|
// FIXME
|
||||||
|
} else if(me.y == startOfLineY) { // single line only processes on itself
|
||||||
int p = me.x - startOfLineX - promptLength + horizontalScrollPosition;
|
int p = me.x - startOfLineX - promptLength + horizontalScrollPosition;
|
||||||
if(p >= 0 && p < line.length) {
|
if(p >= 0 && p < line.length) {
|
||||||
justHitTab = false;
|
justHitTab = false;
|
||||||
|
@ -6897,8 +6981,14 @@ class LineGetter {
|
||||||
this.line[] = line[];
|
this.line[] = line[];
|
||||||
if(cursorPosition > line.length)
|
if(cursorPosition > line.length)
|
||||||
cursorPosition = cast(int) line.length;
|
cursorPosition = cast(int) line.length;
|
||||||
if(horizontalScrollPosition > line.length)
|
if(multiLineMode) {
|
||||||
horizontalScrollPosition = cast(int) line.length;
|
// FIXME?
|
||||||
|
horizontalScrollPosition = 0;
|
||||||
|
verticalScrollPosition = 0;
|
||||||
|
} else {
|
||||||
|
if(horizontalScrollPosition > line.length)
|
||||||
|
horizontalScrollPosition = cast(int) line.length;
|
||||||
|
}
|
||||||
positionCursor();
|
positionCursor();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue