update to new refactor of terminal emulator

This commit is contained in:
Adam D. Ruppe 2020-03-16 21:27:46 -04:00
parent f71c360e83
commit fbbe881537
1 changed files with 38 additions and 261 deletions

View File

@ -6,6 +6,7 @@
+/
module arsd.minigui_addons.terminal_emulator_widget;
///
version(tew_main)
unittest {
import arsd.minigui;
import arsd.minigui_addons.terminal_emulator_widget;
@ -20,6 +21,8 @@ unittest {
auto tew = new TerminalEmulatorWidget([`c:\windows\system32\cmd.exe`], window);
window.loop();
}
main();
}
import arsd.minigui;
@ -27,6 +30,11 @@ import arsd.minigui;
import arsd.terminalemulator;
class TerminalEmulatorWidget : Widget {
this(Widget parent) {
terminalEmulator = new TerminalEmulatorInsideWidget(this);
super(parent);
}
this(string[] args, Widget parent) {
version(Windows) {
import core.sys.windows.windows : HANDLE;
@ -74,7 +82,7 @@ class TerminalEmulatorWidget : Widget {
override MouseCursor cursor() { return GenericCursor.Text; }
override void paint(ScreenPainter painter) {
override void paint(WidgetPainter painter) {
terminalEmulator.redrawPainter(painter, true);
}
}
@ -111,10 +119,7 @@ class TerminalEmulatorInsideWidget : TerminalEmulator {
}
protected override void copyToClipboard(string text) {
static if(UsingSimpledisplayX11)
setPrimarySelection(widget.parentWindow.win, text);
else
setClipboardText(widget.parentWindow.win, text);
setClipboardText(widget.parentWindow.win, text);
}
protected override void pasteFromClipboard(void delegate(in char[]) dg) {
@ -131,6 +136,23 @@ class TerminalEmulatorInsideWidget : TerminalEmulator {
});
}
protected override void copyToPrimary(string text) {
static if(UsingSimpledisplayX11)
setPrimarySelection(widget.parentWindow.win, text);
else
{}
}
protected override void pasteFromPrimary(void delegate(in char[]) dg) {
static if(UsingSimpledisplayX11)
getPrimarySelection(widget.parentWindow.win, dg);
}
override void requestExit() {
// FIXME
}
void resizeImage() { }
mixin PtySupport!(resizeImage);
@ -151,31 +173,15 @@ class TerminalEmulatorInsideWidget : TerminalEmulator {
bool focused;
TerminalEmulatorWidget widget;
OperatingSystemFont font;
mixin SdpyDraw;
private this(TerminalEmulatorWidget widget) {
this.widget = widget;
static if(UsingSimpledisplayX11) {
// FIXME: survive reconnects?
fontSize = 14;
font = new OperatingSystemFont("fixed", fontSize, FontWeight.medium);
if(font.isNull) {
// didn't work, it is using a
// fallback, prolly fixed-13
import std.stdio; writeln("font failed");
fontWidth = 6;
fontHeight = 13;
} else {
fontWidth = fontSize / 2;
fontHeight = fontSize;
}
} else version(Windows) {
font = new OperatingSystemFont("Courier New", fontSize, FontWeight.medium);
fontHeight = fontSize;
fontWidth = fontSize / 2;
}
fontSize = 14;
loadDefaultFont();
auto desiredWidth = 80;
auto desiredHeight = 24;
@ -192,7 +198,8 @@ class TerminalEmulatorInsideWidget : TerminalEmulator {
arsd.terminalemulator.MouseEventType.buttonPressed,
cast(arsd.terminalemulator.MouseButton) ev.button,
(ev.state & ModifierState.shift) ? true : false,
(ev.state & ModifierState.ctrl) ? true : false
(ev.state & ModifierState.ctrl) ? true : false,
(ev.state & ModifierState.alt) ? true : false
))
redraw();
});
@ -205,7 +212,8 @@ class TerminalEmulatorInsideWidget : TerminalEmulator {
arsd.terminalemulator.MouseEventType.buttonReleased,
cast(arsd.terminalemulator.MouseButton) ev.button,
(ev.state & ModifierState.shift) ? true : false,
(ev.state & ModifierState.ctrl) ? true : false
(ev.state & ModifierState.ctrl) ? true : false,
(ev.state & ModifierState.alt) ? true : false
))
redraw();
});
@ -218,7 +226,8 @@ class TerminalEmulatorInsideWidget : TerminalEmulator {
arsd.terminalemulator.MouseEventType.motion,
cast(arsd.terminalemulator.MouseButton) ev.button,
(ev.state & ModifierState.shift) ? true : false,
(ev.state & ModifierState.ctrl) ? true : false
(ev.state & ModifierState.ctrl) ? true : false,
(ev.state & ModifierState.alt) ? true : false
))
redraw();
});
@ -298,21 +307,15 @@ class TerminalEmulatorInsideWidget : TerminalEmulator {
}
}
int fontWidth;
int fontHeight;
static int fontSize = 14;
enum paddingLeft = 2;
enum paddingTop = 1;
bool clearScreenRequested = true;
void redraw(bool forceRedraw = false) {
if(widget.parentWindow is null || widget.parentWindow.win is null)
return;
auto painter = widget.draw();
if(clearScreenRequested) {
auto clearColor = defaultTextAttributes.background;
auto clearColor = defaultBackground;
painter.outlineColor = clearColor;
painter.fillColor = clearColor;
painter.drawRectangle(Point(0, 0), widget.width, widget.height);
@ -323,231 +326,5 @@ class TerminalEmulatorInsideWidget : TerminalEmulator {
redrawPainter(painter, forceRedraw);
}
bool lastDrawAlternativeScreen;
final arsd.color.Rectangle redrawPainter(T)(T painter, bool forceRedraw) {
arsd.color.Rectangle invalidated;
// FIXME: could prolly use optimizations
painter.setFont(font);
int posx = paddingLeft;
int posy = paddingTop;
char[512] bufferText;
bool hasBufferedInfo;
int bufferTextLength;
Color bufferForeground;
Color bufferBackground;
int bufferX = -1;
int bufferY = -1;
bool bufferReverse;
void flushBuffer() {
if(!hasBufferedInfo) {
return;
}
assert(posx - bufferX - 1 > 0);
painter.fillColor = bufferReverse ? bufferForeground : bufferBackground;
painter.outlineColor = bufferReverse ? bufferForeground : bufferBackground;
painter.drawRectangle(Point(bufferX, bufferY), posx - bufferX, fontHeight);
painter.fillColor = Color.transparent;
// Hack for contrast!
if(bufferBackground == Color.black && !bufferReverse) {
// brighter than normal in some cases so i can read it easily
painter.outlineColor = contrastify(bufferForeground);
} else if(bufferBackground == Color.white && !bufferReverse) {
// darker than normal so i can read it
painter.outlineColor = antiContrastify(bufferForeground);
} else if(bufferForeground == bufferBackground) {
// color on itself, I want it visible too
auto hsl = toHsl(bufferForeground, true);
if(hsl[2] < 0.5)
hsl[2] += 0.5;
else
hsl[2] -= 0.5;
painter.outlineColor = fromHsl(hsl[0], hsl[1], hsl[2]);
} else {
// normal
painter.outlineColor = bufferReverse ? bufferBackground : bufferForeground;
}
// FIXME: make sure this clips correctly
painter.drawText(Point(bufferX, bufferY), cast(immutable) bufferText[0 .. bufferTextLength]);
hasBufferedInfo = false;
bufferReverse = false;
bufferTextLength = 0;
bufferX = -1;
bufferY = -1;
}
int x;
foreach(idx, ref cell; alternateScreenActive ? alternateScreen : normalScreen) {
if(!forceRedraw && !cell.invalidated && lastDrawAlternativeScreen == alternateScreenActive) {
flushBuffer();
goto skipDrawing;
}
cell.invalidated = false;
version(none) if(bufferX == -1) { // why was this ever here?
bufferX = posx;
bufferY = posy;
}
{
invalidated.left = posx < invalidated.left ? posx : invalidated.left;
invalidated.top = posy < invalidated.top ? posy : invalidated.top;
int xmax = posx + fontWidth;
int ymax = posy + fontHeight;
invalidated.right = xmax > invalidated.right ? xmax : invalidated.right;
invalidated.bottom = ymax > invalidated.bottom ? ymax : invalidated.bottom;
// FIXME: this could be more efficient, simpledisplay could get better graphics context handling
{
bool reverse = (cell.attributes.inverse != reverseVideo);
if(cell.selected)
reverse = !reverse;
auto fgc = cell.attributes.foreground;
auto bgc = cell.attributes.background;
if(!(cell.attributes.foregroundIndex & 0xff00)) {
// this refers to a specific palette entry, which may change, so we should use that
fgc = palette[cell.attributes.foregroundIndex];
}
if(!(cell.attributes.backgroundIndex & 0xff00)) {
// this refers to a specific palette entry, which may change, so we should use that
bgc = palette[cell.attributes.backgroundIndex];
}
if(fgc != bufferForeground || bgc != bufferBackground || reverse != bufferReverse)
flushBuffer();
bufferReverse = reverse;
bufferBackground = bgc;
bufferForeground = fgc;
}
}
if(cell.ch != dchar.init) {
char[4] str;
import std.utf;
// now that it is buffered, we do want to draw it this way...
//if(cell.ch != ' ') { // no point wasting time drawing spaces, which are nothing; the bg rectangle already did the important thing
try {
auto stride = encode(str, cell.ch);
if(bufferTextLength + stride > bufferText.length)
flushBuffer();
bufferText[bufferTextLength .. bufferTextLength + stride] = str[0 .. stride];
bufferTextLength += stride;
if(bufferX == -1) {
bufferX = posx;
bufferY = posy;
}
hasBufferedInfo = true;
} catch(Exception e) {
import std.stdio;
writeln(cast(uint) cell.ch, " :: ", e.msg);
}
//}
} else if(cell.nonCharacterData !is null) {
}
if(cell.attributes.underlined) {
// the posx adjustment is because the buffer assumes it is going
// to be flushed after advancing, but here, we're doing it mid-character
// FIXME: we should just underline the whole thing consecutively, with the buffer
posx += fontWidth;
flushBuffer();
posx -= fontWidth;
painter.drawLine(Point(posx, posy + fontHeight - 1), Point(posx + fontWidth, posy + fontHeight - 1));
}
skipDrawing:
posx += fontWidth;
x++;
if(x == screenWidth) {
flushBuffer();
x = 0;
posy += fontHeight;
posx = paddingLeft;
}
}
if(cursorShowing) {
painter.fillColor = cursorColor;
painter.outlineColor = cursorColor;
painter.rasterOp = RasterOp.xor;
posx = cursorPosition.x * fontWidth + paddingLeft;
posy = cursorPosition.y * fontHeight + paddingTop;
int cursorWidth = fontWidth;
int cursorHeight = fontHeight;
final switch(cursorStyle) {
case CursorStyle.block:
painter.drawRectangle(Point(posx, posy), cursorWidth, cursorHeight);
break;
case CursorStyle.underline:
painter.drawRectangle(Point(posx, posy + cursorHeight - 2), cursorWidth, 2);
break;
case CursorStyle.bar:
painter.drawRectangle(Point(posx, posy), 2, cursorHeight);
break;
}
painter.rasterOp = RasterOp.normal;
// since the cursor draws over the cell, we need to make sure it is redrawn each time too
auto buffer = alternateScreenActive ? (&alternateScreen) : (&normalScreen);
if(cursorX >= 0 && cursorY >= 0 && cursorY < screenHeight && cursorX < screenWidth) {
(*buffer)[cursorY * screenWidth + cursorX].invalidated = true;
}
invalidated.left = posx < invalidated.left ? posx : invalidated.left;
invalidated.top = posy < invalidated.top ? posy : invalidated.top;
int xmax = posx + fontWidth;
int ymax = xmax + fontHeight;
invalidated.right = xmax > invalidated.right ? xmax : invalidated.right;
invalidated.bottom = ymax > invalidated.bottom ? ymax : invalidated.bottom;
}
lastDrawAlternativeScreen = alternateScreenActive;
return invalidated;
}
// black bg, make the colors more visible
Color contrastify(Color c) {
if(c == Color(0xcd, 0, 0))
return Color.fromHsl(0, 1.0, 0.75);
else if(c == Color(0, 0, 0xcd))
return Color.fromHsl(240, 1.0, 0.75);
else if(c == Color(229, 229, 229))
return Color(0x99, 0x99, 0x99);
else return c;
}
// white bg, make them more visible
Color antiContrastify(Color c) {
if(c == Color(0xcd, 0xcd, 0))
return Color.fromHsl(60, 1.0, 0.25);
else if(c == Color(0, 0xcd, 0xcd))
return Color.fromHsl(180, 1.0, 0.25);
else if(c == Color(229, 229, 229))
return Color(0x99, 0x99, 0x99);
else return c;
}
bool debugMode = false;
}