From 104adea1efaeb468bdbae7b045fdfcb71650434a Mon Sep 17 00:00:00 2001 From: "Adam D. Ruppe" Date: Thu, 7 Dec 2017 10:41:28 -0500 Subject: [PATCH 1/2] new text thing --- simpledisplay.d | 227 ++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 227 insertions(+) diff --git a/simpledisplay.d b/simpledisplay.d index 16f353a..5a248a8 100644 --- a/simpledisplay.d +++ b/simpledisplay.d @@ -11494,6 +11494,233 @@ version(X11) { } } +mixin template ExperimentalTextComponent2() { + + enum TextFormat : ushort { + // decorations + underline = 1, + strikethrough = 2, + + // font selectors + + bold = 0x4000 | 1, // weight 700 + light = 0x4000 | 2, // weight 300 + veryBoldOrLight = 0x4000 | 4, // weight 100 with light, weight 900 with bold + // bold | light is really invalid but should give weight 500 + // veryBoldOrLight without one of the others should just give the default for the font; it should be ignored. + + italic = 0x4000 | 8, + smallcaps = 0x4000 | 16, + } + + + struct Decoration { + ushort id; + Color foreground; + Color background; + ushort textFormat; + void* font; + } + + Decoration[] decorations; + + struct TextState { + char[] text; + int[] x; + int[] y; + ushort[] decorationId; + int length; + + int caret; + + void makeGap(int where, int minLength) { + int gapSize = 0; + int at = where; + while(at < text.length && text[at] == 0xff) { + at++; + gapSize++; + } + + if(gapSize >= minLength) + return; + + // try to gather gap from behind us, if any + /* + at = where - 32; + if(at < 0) + at = 0; + + while(at < where - 1) { + if(text[at] == 0xff) { + text[at] = text[at + 1]; + x[at] = x[at + 1]; + y[at] = y[at + 1]; + decorationId[at] = decorationId[at + 1]; + text[at + 1] = 0xff; + gapSize++; + } + at++; + } + + if(gapSize >= minLength) + return; + */ + keep_trying: + at = where; + while(at + 1 < text.length) { + // FIXME it needs to work on a whole block, not just one char + if(text[at + 1] == 0xff) { + text[at + 1] = text[at]; + x[at + 1] = x[at]; + y[at + 1] = y[at]; + decorationId[at + 1] = decorationId[at]; + text[at] = 0xff; + gapSize++; + if(gapSize >= minLength) + return; + } + at++; + } + + if(gapSize < minLength) { + auto increase = 16; + if(minLength - gapSize > 16) + increase = minLength - gapSize; + text.length += increase; + x.length += increase; + y.length += increase; + decorationId.length += increase; + text[$ - increase .. $] = 0xff; + goto keep_trying; + } + } + + void insert(dchar c) { + makeGap(caret, 1); + text[caret] = cast(char) c; + caret++; + length++; + layout(caret - 1, cast(int) text.length, false); + } + + string toPlainText() { + string s; + s.reserve(length); + foreach(char ch; text) + if(ch != 0xff) + s ~= ch; + return s; + } + + void resetContents(in char[] to) { + if(text.length < to.length * 2) { + text.length = to.length * 2; + x.length = text.length; + y.length = text.length; + decorationId.length = text.length; + } + int textPos = 0; + int skipped = 0; + foreach(ch; to) { + if(ch == 13) { + ++skipped; + continue; + } + text[textPos++] = ch; + text[textPos++] = 0xff; + } + text[textPos .. $] = 0xff; + length = cast(int) to.length - skipped; + + decorationId[0 .. length] = 0; + + layout(0, text.length, true); + } + + int lineHeight = 14; + int letterWidth = 7; + int tabStop = 4; + + void layout(int start, int end, bool forceAll) { + int x = 0, y = 0; + foreach(idx, char ch; text[start .. end]) { + if(ch == 0xff) + continue; + if(!forceAll && this.x[start + idx] == x && this.y[start + idx] == y) + break; // seems to already be done! + this.x[start + idx] = x; + this.y[start + idx] = y; + + // FIXME unicode + + if(ch == '\n') { + x = 0; + y += lineHeight; + } else if(ch == '\t') { + x += x % (letterWidth * tabStop); + } else { + x += letterWidth; + } + } + } + + void drawInto(ScreenPainter painter, int dx, int dy, int sx, int sy, int width, int height) { + //char[6] buffer; + // FIXME unicode + + painter.outlineColor = Color.white; + painter.fillColor = Color.white; + painter.drawRectangle(Point(dx, dy), width, height); + + painter.outlineColor = Color.black; + + if(length == 0) + return; + + int startingIdx = 0; + if(sx > 0 || sy > 0) { + int lastSearched = text.length; + // binary search till we get the first visible item + startingIdx = text.length / 2; + keep_searching: + while(startingIdx >= 0 && text[startingIdx] == 0xff) + --startingIdx; + while(text[startingIdx] == 0xff && startingIdx < text.length) + ++startingIdx; + if(startingIdx == text.length) + assert(0); // we're apparently empty! why didn't length == 0? + + if(this.x[startingIdx] > sx || this.y[startingIdx] > sy) { + // FIXME + // too far ahead, search backward + lastSearched = startingIdx; + startingIdx = startingIdx / 2; + goto keep_searching; + } else { + // this is probably good enough but let's try to be more precise + //startingIdx = (lastSearched - startingIdx) / 2; + //goto keep_searching; + } + } + + foreach(idx, char ch; text[startingIdx .. $]) { + if(ch == 0xff) + continue; + int drawX = dx + this.x[startingIdx + idx] - sx; + int drawY = dy + this.y[startingIdx + idx] - sy; + + if(drawX - dx > width) + continue; + if(drawY - dy > height) + break; + + painter.drawText(Point(drawX, drawY), "" ~ ch); + import std.stdio; write(ch); stdout.flush; + } + } + } +} + // Don't use this yet. When I'm happy with it, I will move it to the // regular module namespace. From 47bacf04562bf002d7ab47dc2d3518780b8b11fb Mon Sep 17 00:00:00 2001 From: "H. S. Teoh" Date: Tue, 12 Dec 2017 17:58:12 -0800 Subject: [PATCH 2/2] Range API needs size_t for .length. --- database.d | 2 +- mssql.d | 2 +- mysql.d | 2 +- postgres.d | 2 +- sqlite.d | 4 ++-- 5 files changed, 6 insertions(+), 6 deletions(-) diff --git a/database.d b/database.d index 639589c..c4e2632 100644 --- a/database.d +++ b/database.d @@ -147,7 +147,7 @@ interface ResultSet { bool empty() @property; Row front() @property; void popFront() ; - int length() @property; + size_t length() @property; /* deprecated */ final ResultSet byAssoc() { return this; } } diff --git a/mssql.d b/mssql.d index ba86d55..8957bd8 100644 --- a/mssql.d +++ b/mssql.d @@ -111,7 +111,7 @@ class MsSqlResult : ResultSet { fetchNext; } - int length() + override size_t length() { return 1; //FIX ME } diff --git a/mysql.d b/mysql.d index 73ad5a6..aa59bdd 100644 --- a/mysql.d +++ b/mysql.d @@ -72,7 +72,7 @@ class MySqlResult : ResultSet { } - override int length() { + override size_t length() { if(result is null) return 0; return cast(int) mysql_num_rows(result); diff --git a/postgres.d b/postgres.d index bc87a8d..6c396d9 100644 --- a/postgres.d +++ b/postgres.d @@ -126,7 +126,7 @@ class PostgresResult : ResultSet { fetchNext(); } - int length() { + override size_t length() { return numRows; } diff --git a/sqlite.d b/sqlite.d index 90fad9e..f7e6b11 100644 --- a/sqlite.d +++ b/sqlite.d @@ -227,8 +227,8 @@ class SqliteResult : ResultSet { position++; } - int length() { - return cast(int) rows.length; + override size_t length() { + return rows.length; } this(Variant[][] rows, char[][] columnNames) {