mirror of https://github.com/adamdruppe/arsd.git
update to new refactor of terminal emulator
This commit is contained in:
parent
f71c360e83
commit
fbbe881537
|
@ -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;
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue