From f65108dbea54ce899d42c7287b28ac701086e631 Mon Sep 17 00:00:00 2001 From: Vadim Lopatin Date: Mon, 30 May 2016 10:51:56 +0300 Subject: [PATCH] terminal - initial implementation --- dlangide_msvc.visualdproj | 17 +- src/dlangide/ui/outputpanel.d | 13 ++ src/dlangide/ui/terminal.d | 305 ++++++++++++++++++++++++++++++++ views/res/ide_theme_default.xml | 4 + 4 files changed, 331 insertions(+), 8 deletions(-) create mode 100644 src/dlangide/ui/terminal.d diff --git a/dlangide_msvc.visualdproj b/dlangide_msvc.visualdproj index 9f75d90..fbedae6 100644 --- a/dlangide_msvc.visualdproj +++ b/dlangide_msvc.visualdproj @@ -53,7 +53,7 @@ $(CC) -c 1 $(DMDInstallDir)windows\bin\dmd.exe - $(ProjectDir)/../dlangui/src $(ProjectDir)/../dlangui/3rdparty $(ProjectDir)/../dlangui/deps/DerelictGL3/source $(ProjectDir)/../dlangui/deps/DerelictUtil/source $(ProjectDir)/../dlangui/deps/DerelictFT/source $(ProjectDir)/../dlangui/deps/DerelictSDL2/source $(ProjectDir)/../dlangui/deps/libdparse/src + $(ProjectDir)/../dlangui/src $(ProjectDir)/../dlangui/3rdparty $(ProjectDir)/../dlangui/deps/DerelictGL3/source $(ProjectDir)/../dlangui/deps/DerelictUtil/source $(ProjectDir)/../dlangui/deps/DerelictFT/source $(ProjectDir)/../dlangui/deps/DerelictSDL2/source $(ProjectDir)/../dlangui/deps/libdparse/src $(ProjectDir)/../DerelictLLDB/source views views/res views/res/i18n views/res/mdpi views/res/hdpi $(ConfigurationName) $(OutDir) @@ -72,7 +72,7 @@ 0 DebugInfo DCD 0 - EmbedStandardResources USE_FREETYPE + EmbedStandardResources 0 0 0 @@ -155,7 +155,7 @@ $(CC) -c 1 $(DMDInstallDir)windows\bin\dmd.exe - $(ProjectDir)/../dlangui/src $(ProjectDir)/../dlangui/3rdparty $(ProjectDir)/../dlangui/deps/DerelictGL3/source $(ProjectDir)/../dlangui/deps/DerelictUtil/source $(ProjectDir)/../dlangui/deps/DerelictFT/source $(ProjectDir)/../dlangui/deps/DerelictSDL2/source $(ProjectDir)/../dlangui/deps/libdparse/src + $(ProjectDir)/../dlangui/src $(ProjectDir)/../dlangui/3rdparty $(ProjectDir)/../dlangui/deps/DerelictGL3/source $(ProjectDir)/../dlangui/deps/DerelictUtil/source $(ProjectDir)/../dlangui/deps/DerelictFT/source $(ProjectDir)/../dlangui/deps/DerelictSDL2/source $(ProjectDir)/../dlangui/deps/libdparse/src $(ProjectDir)/../DerelictLLDB/source views views/res views/res/i18n views/res/mdpi views/res/hdpi $(ConfigurationName) $(OutDir) @@ -212,7 +212,7 @@ 0 0 0 - 0 + 1 0 0 0 @@ -254,10 +254,10 @@ 0 0 0 - $(CC) -c + $(CC) -c -v 1 $(DMDInstallDir)windows\bin\dmd.exe - $(ProjectDir)/../dlangui/src $(ProjectDir)/../dlangui/3rdparty $(ProjectDir)/../dlangui/deps/DerelictGL3/source $(ProjectDir)/../dlangui/deps/DerelictUtil/source $(ProjectDir)/../dlangui/deps/DerelictFT/source $(ProjectDir)/../dlangui/deps/DerelictSDL2/source $(ProjectDir)/../dlangui/deps/libdparse/src + $(ProjectDir)/../dlangui/src $(ProjectDir)/../dlangui/3rdparty $(ProjectDir)/../dlangui/deps/DerelictGL3/source $(ProjectDir)/../dlangui/deps/DerelictUtil/source $(ProjectDir)/../dlangui/deps/DerelictFT/source $(ProjectDir)/../dlangui/deps/DerelictSDL2/source $(ProjectDir)/../dlangui/deps/libdparse/src $(ProjectDir)/../DerelictLLDB/source views views/res views/res/i18n views/res/mdpi views/res/hdpi $(ConfigurationName) $(OutDir) @@ -300,7 +300,7 @@ $(OutDir)\$(ProjectName).exe 1 2 - 0 + 1 @@ -359,7 +359,7 @@ $(CC) -c 1 $(DMDInstallDir)windows\bin\dmd.exe - $(ProjectDir)/../dlangui/src $(ProjectDir)/../dlangui/3rdparty $(ProjectDir)/../dlangui/deps/DerelictGL3/source $(ProjectDir)/../dlangui/deps/DerelictUtil/source $(ProjectDir)/../dlangui/deps/DerelictFT/source $(ProjectDir)/../dlangui/deps/DerelictSDL2/source $(ProjectDir)/../dlangui/deps/libdparse/src + $(ProjectDir)/../dlangui/src $(ProjectDir)/../dlangui/3rdparty $(ProjectDir)/../dlangui/deps/DerelictGL3/source $(ProjectDir)/../dlangui/deps/DerelictUtil/source $(ProjectDir)/../dlangui/deps/DerelictFT/source $(ProjectDir)/../dlangui/deps/DerelictSDL2/source $(ProjectDir)/../dlangui/deps/libdparse/src $(ProjectDir)/../DerelictLLDB/source views views/res views/res/i18n views/res/mdpi views/res/hdpi $(ConfigurationName) $(OutDir) @@ -540,6 +540,7 @@ + diff --git a/src/dlangide/ui/outputpanel.d b/src/dlangide/ui/outputpanel.d index f5dd49d..73a0aa8 100644 --- a/src/dlangide/ui/outputpanel.d +++ b/src/dlangide/ui/outputpanel.d @@ -4,6 +4,7 @@ import dlangui; import dlangide.workspace.workspace; import dlangide.workspace.project; import dlangide.ui.frame; +import dlangide.ui.terminal; import std.utf; import std.regex; @@ -125,6 +126,7 @@ class OutputPanel : DockWindow { Signal!CompilerLogIssueClickHandler compilerLogIssueClickHandler; protected CompilerLogWidget _logWidget; + protected TerminalWidget _terminalWidget; TabWidget _tabs; @@ -154,6 +156,17 @@ class OutputPanel : DockWindow { _tabs.addTab(_logWidget, "Compiler Log"d); _tabs.selectTab("logwidget"); + _terminalWidget = new TerminalWidget("TERMINAL"); + _terminalWidget.layoutWidth(FILL_PARENT).layoutHeight(FILL_PARENT); + _tabs.addTab(_terminalWidget, "Output"d); + _terminalWidget.write("Hello\nSecond line\nTest\n"d); + _terminalWidget.write("SomeString 123456789\rAwesomeString\n"d); // test \r + _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); + return _tabs; } diff --git a/src/dlangide/ui/terminal.d b/src/dlangide/ui/terminal.d new file mode 100644 index 0000000..93b00c3 --- /dev/null +++ b/src/dlangide/ui/terminal.d @@ -0,0 +1,305 @@ +module dlangide.ui.terminal; + +import dlangui.widgets.widget; +import dlangui.widgets.controls; + +struct TerminalChar { + ubyte bgColor = 0; + ubyte textColor = 0; + dchar ch = ' '; +} + +struct TerminalLine { + TerminalChar[] line; + bool overflowFlag; + bool eolFlag; + void clear() { + line.length = 0; + overflowFlag = false; + eolFlag = false; + } + void markLineOverflow() {} + void markLineEol() {} + void putCharAt(dchar ch, int x) { + if (x >= line.length) { + while (x >= line.length) { + line.assumeSafeAppend; + line ~= TerminalChar.init; + } + } + line[x].ch = ch; + } +} + +struct TerminalContent { + TerminalLine[] lines; + Rect rc; + FontRef font; + int maxBufferLines = 3000; + int topLine; + int width; // width in chars + int height; // height in chars + int charw; // single char width + int charh; // single char height + int cursorx; + int cursory; + void layout(FontRef font, Rect rc) { + this.rc = rc; + this.font = font; + this.charw = font.charWidth('0'); + this.charh = font.height; + int w = rc.width / charw; + int h = rc.height / charh; + setViewSize(w, h); + } + void setViewSize(int w, int h) { + if (h < 2) + h = 2; + if (w < 16) + w = 16; + width = w; + height = h; + } + void draw(DrawBuf buf) { + Rect lineRect = rc; + dchar[] text; + text.length = 1; + text[0] = ' '; + for (uint i = 0; i < height && i + topLine < lines.length; i++) { + lineRect.bottom = lineRect.top + charh; + TerminalLine * p = &lines[i + topLine]; + // draw line in rect + for (int x = 0; x < p.line.length; x++) { + dchar ch = p.line[x].ch; + if (ch >= ' ') { + text[0] = ch; + font.drawText(buf, lineRect.left + x * charw, lineRect.top, text, 0); + } + } + 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; + 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; + } + } + if (cast(uint)y >= lines.length) { + for (auto i = lines.length; i <= y; i++) + lines ~= TerminalLine.init; + } + return &lines[y]; + } + void putCharAt(dchar ch, ref int x, ref int y) { + if (x < 0) + x = 0; + TerminalLine * line = getLine(y); + if (x >= width) { + line.markLineOverflow(); + y++; + line = getLine(y); + x = 0; + } + line.putCharAt(ch, x); + } + int tabSize = 8; + // supports printed characters and \r \n \t + void putChar(dchar ch) { + if (ch == '\r') { + cursorx = 0; + return; + } + if (ch == '\n') { + TerminalLine * line = getLine(cursory); + line.markLineEol(); + cursory++; + line = getLine(cursory); + cursorx = 0; + return; + } + if (ch == '\t') { + int newx = (cursorx + tabSize) / tabSize * tabSize; + if (newx > width) { + TerminalLine * line = getLine(cursory); + line.markLineEol(); + cursory++; + line = getLine(cursory); + cursorx = 0; + } else { + for (int x = cursorx; x < newx; x++) { + putCharAt(' ', cursorx, cursory); + cursorx++; + } + } + return; + } + putCharAt(ch, cursorx, cursory); + cursorx++; + } + + void updateScrollBar(ScrollBar sb) { + sb.pageSize = height; + sb.maxValue = lines.length ? lines.length - 1 : 0; + sb.position = topLine; + } +} + +class TerminalWidget : WidgetGroup { + protected ScrollBar _verticalScrollBar; + protected TerminalContent _content; + this() { + this(null); + } + this(string ID) { + super(ID); + styleId = "TERMINAL"; + _verticalScrollBar = new ScrollBar("VERTICAL_SCROLLBAR", Orientation.Vertical); + _verticalScrollBar.minValue = 0; + addChild(_verticalScrollBar); + } + /** + Measure widget according to desired width and height constraints. (Step 1 of two phase layout). + + */ + override void measure(int parentWidth, int parentHeight) { + int w = (parentWidth == SIZE_UNSPECIFIED) ? font.charWidth('0') * 80 : parentWidth; + int h = (parentHeight == SIZE_UNSPECIFIED) ? font.height * 10 : parentHeight; + Rect rc = Rect(0, 0, w, h); + applyMargins(rc); + applyPadding(rc); + _verticalScrollBar.measure(rc.width, rc.height); + rc.right -= _verticalScrollBar.measuredWidth; + measuredContent(parentWidth, parentHeight, rc.width, rc.height); + } + + /// Set widget rectangle to specified value and layout widget contents. (Step 2 of two phase layout). + override void layout(Rect rc) { + if (visibility == Visibility.Gone) { + return; + } + _pos = rc; + _needLayout = false; + applyMargins(rc); + applyPadding(rc); + Rect sbrc = rc; + sbrc.left = sbrc.right - _verticalScrollBar.measuredWidth; + _verticalScrollBar.layout(sbrc); + rc.right = sbrc.left; + _content.layout(font, rc); + if (outputChars.length) { + // push buffered text + write(""d); + _needLayout = false; + } + } + /// Draw widget at its position to buffer + override void onDraw(DrawBuf buf) { + if (visibility != Visibility.Visible) + return; + Rect rc = _pos; + applyMargins(rc); + auto saver = ClipRectSaver(buf, rc, alpha); + DrawableRef bg = backgroundDrawable; + if (!bg.isNull) { + bg.drawTo(buf, rc, state); + } + applyPadding(rc); + _verticalScrollBar.onDraw(buf); + _content.draw(buf); + } + + private char[] outputBuffer; + // write utf 8 + void write(string bytes) { + if (!bytes.length) + return; + import std.utf; + outputBuffer.assumeSafeAppend; + outputBuffer ~= bytes; + size_t index = 0; + dchar[] decoded; + decoded.assumeSafeAppend; + dchar ch = 0; + for (;;) { + size_t oldindex = index; + try { + ch = decode(outputBuffer, index); + decoded ~= ch; + } catch (UTFException e) { + if (index + 4 <= outputBuffer.length) { + // just append invalid character + ch = '?'; + index++; + } + } + if (oldindex == index) + break; + } + if (index > 0) { + // move content + for (size_t i = 0; i + index < outputBuffer.length; i++) + outputBuffer[i] = outputBuffer[i + index]; + outputBuffer.length = outputBuffer.length - index; + } + if (decoded.length) + write(cast(dstring)decoded); + } + + private dchar[] outputChars; + // write utf32 + void write(dstring chars) { + if (!chars.length && !outputChars.length) + return; + outputChars.assumeSafeAppend; + outputChars ~= chars; + if (!_content.width) + return; + uint i = 0; + for (; i < outputChars.length; i++) { + int ch = outputChars[i]; + if (ch < ' ') { + // control character + switch(ch) { + case '\r': + case '\n': + case '\t': + _content.putChar(ch); + break; + default: + break; + } + } else { + _content.putChar(ch); + } + } + if (i > 0) { + if (i == outputChars.length) + outputChars.length = 0; + else { + for (uint j = 0; j + i < outputChars.length; j++) + outputChars[j] = outputChars[j + i]; + outputChars.length = outputChars.length - i; + } + } + _content.updateScrollBar(_verticalScrollBar); + } +} diff --git a/views/res/ide_theme_default.xml b/views/res/ide_theme_default.xml index 1c4db62..b06fbaa 100644 --- a/views/res/ide_theme_default.xml +++ b/views/res/ide_theme_default.xml @@ -18,5 +18,9 @@ +