more text improvements

This commit is contained in:
Adam D. Ruppe 2017-04-06 10:36:17 -04:00
parent 9deb9d0801
commit 11a5c1e43f
3 changed files with 263 additions and 133 deletions

View File

@ -306,6 +306,8 @@ class FreeEntrySelection : ComboboxBase {
tabStop = false; tabStop = false;
lineEdit.addEventListener("focus", &lineEdit.selectAll);
auto btn = new class ArrowButton { auto btn = new class ArrowButton {
this() { this() {
super(ArrowDirection.down, hl); super(ArrowDirection.down, hl);
@ -365,6 +367,8 @@ class ComboBox : ComboboxBase {
redraw(); redraw();
}); });
lineEdit.addEventListener("focus", &lineEdit.selectAll);
listWidget.addDirectEventListener(EventType.change, { listWidget.addDirectEventListener(EventType.change, {
int set = -1; int set = -1;
foreach(idx, opt; listWidget.options) foreach(idx, opt; listWidget.options)
@ -515,13 +519,13 @@ class DataView : Widget {
/* /*
TextEdit needs: TextEdit needs:
* carat manipulation * caret manipulation
* selection control * selection control
* convenience functions for appendText, insertText, insertTextAtCarat, etc. * convenience functions for appendText, insertText, insertTextAtCaret, etc.
For example: For example:
connect(paste, &textEdit.insertTextAtCarat); connect(paste, &textEdit.insertTextAtCaret);
would be nice. would be nice.
@ -1895,6 +1899,8 @@ class HorizontalLayout : Layout {
int lastMargin = 0; int lastMargin = 0;
foreach(child; children) { foreach(child; children) {
auto mh = child.maxHeight(); auto mh = child.maxHeight();
if(mh == int.max)
return int.max;
if(mh > largest) if(mh > largest)
largest = mh; largest = mh;
margins += mymax(lastMargin, child.marginTop()); margins += mymax(lastMargin, child.marginTop());
@ -1908,7 +1914,7 @@ class HorizontalLayout : Layout {
foreach(child; children) { foreach(child; children) {
auto c = child.heightStretchiness; auto c = child.heightStretchiness;
if(c > max) if(c > max)
c = max; max = c;
} }
return max; return max;
} }
@ -2384,6 +2390,10 @@ class LabeledLineEdit : Widget {
void content(string c) { void content(string c) {
return lineEdit.content(c); return lineEdit.content(c);
} }
void selectAll() {
lineEdit.selectAll();
}
} }
/// ///
@ -3563,6 +3573,15 @@ abstract class EditableTextWidget : ScrollableWidget {
override int minHeight() { return Window.lineHeight + 0; } // the +0 is to leave room for the padding override int minHeight() { return Window.lineHeight + 0; } // the +0 is to leave room for the padding
override int widthStretchiness() { return 7; } override int widthStretchiness() { return 7; }
void selectAll() {
version(win32_widgets)
SendMessage(hwnd, EM_SETSEL, 0, -1);
else version(custom_widgets) {
textLayout.selectAll();
redraw();
}
}
@property string content() { @property string content() {
version(win32_widgets) { version(win32_widgets) {
char[4096] buffer; char[4096] buffer;
@ -3602,7 +3621,7 @@ abstract class EditableTextWidget : ScrollableWidget {
else version(custom_widgets) { else version(custom_widgets) {
// FIXME // FIXME
Timer caratTimer; Timer caretTimer;
TextLayout textLayout; TextLayout textLayout;
void setupCustomTextEditing() { void setupCustomTextEditing() {
@ -3616,38 +3635,38 @@ abstract class EditableTextWidget : ScrollableWidget {
painter.outlineColor = Color.black; painter.outlineColor = Color.black;
// painter.drawText(Point(4, 4), content, Point(width - 4, height - 4)); // painter.drawText(Point(4, 4), content, Point(width - 4, height - 4));
textLayout.caratShowingOnScreen = false; textLayout.caretShowingOnScreen = false;
textLayout.drawInto(painter, !parentWindow.win.closed && isFocused()); textLayout.drawInto(painter, !parentWindow.win.closed && isFocused());
}; };
defaultEventHandlers["click"] = delegate (Widget _this, Event ev) { defaultEventHandlers["click"] = delegate (Widget _this, Event ev) {
if(parentWindow.win.closed) return; if(parentWindow.win.closed) return;
textLayout.moveCaratToPixelCoordinates(ev.clientX, ev.clientY); textLayout.moveCaretToPixelCoordinates(ev.clientX, ev.clientY);
this.focus(); this.focus();
}; };
defaultEventHandlers["focus"] = delegate (Widget _this, Event ev) { defaultEventHandlers["focus"] = delegate (Widget _this, Event ev) {
if(parentWindow.win.closed) return; if(parentWindow.win.closed) return;
auto painter = this.draw(); auto painter = this.draw();
textLayout.drawCarat(painter); textLayout.drawCaret(painter);
if(caratTimer) { if(caretTimer) {
caratTimer.destroy(); caretTimer.destroy();
caratTimer = null; caretTimer = null;
} }
caratTimer = new Timer(500, { caretTimer = new Timer(500, {
if(parentWindow.win.closed) { if(parentWindow.win.closed) {
caratTimer.destroy(); caretTimer.destroy();
return; return;
} }
if(isFocused()) { if(isFocused()) {
auto painter = this.draw(); auto painter = this.draw();
textLayout.drawCarat(painter); textLayout.drawCaret(painter);
} else if(textLayout.caratShowingOnScreen) { } else if(textLayout.caretShowingOnScreen) {
auto painter = this.draw(); auto painter = this.draw();
textLayout.eraseCarat(painter); textLayout.eraseCaret(painter);
} }
}); });
@ -3655,10 +3674,10 @@ abstract class EditableTextWidget : ScrollableWidget {
defaultEventHandlers["blur"] = delegate (Widget _this, Event ev) { defaultEventHandlers["blur"] = delegate (Widget _this, Event ev) {
if(parentWindow.win.closed) return; if(parentWindow.win.closed) return;
auto painter = this.draw(); auto painter = this.draw();
textLayout.eraseCarat(painter); textLayout.eraseCaret(painter);
if(caratTimer) { if(caretTimer) {
caratTimer.destroy(); caretTimer.destroy();
caratTimer = null; caretTimer = null;
} }
auto evt = new Event(EventType.change, this); auto evt = new Event(EventType.change, this);
@ -3680,27 +3699,27 @@ abstract class EditableTextWidget : ScrollableWidget {
redraw(); redraw();
break; break;
case Key.Left: case Key.Left:
textLayout.moveLeft(textLayout.carat); textLayout.moveLeft(textLayout.caret);
redraw(); redraw();
break; break;
case Key.Right: case Key.Right:
textLayout.moveRight(textLayout.carat); textLayout.moveRight(textLayout.caret);
redraw(); redraw();
break; break;
case Key.Up: case Key.Up:
textLayout.moveUp(textLayout.carat); textLayout.moveUp(textLayout.caret);
redraw(); redraw();
break; break;
case Key.Down: case Key.Down:
textLayout.moveDown(textLayout.carat); textLayout.moveDown(textLayout.caret);
redraw(); redraw();
break; break;
case Key.Home: case Key.Home:
textLayout.moveHome(textLayout.carat); textLayout.moveHome(textLayout.caret);
redraw(); redraw();
break; break;
case Key.End: case Key.End:
textLayout.moveEnd(textLayout.carat); textLayout.moveEnd(textLayout.caret);
redraw(); redraw();
break; break;
default: default:
@ -3724,8 +3743,10 @@ abstract class EditableTextWidget : ScrollableWidget {
/// ///
class LineEdit : EditableTextWidget { class LineEdit : EditableTextWidget {
// FIXME: hack // FIXME: hack
version(custom_widgets) {
override bool showingVerticalScroll() { return false; } override bool showingVerticalScroll() { return false; }
override bool showingHorizontalScroll() { return false; } override bool showingHorizontalScroll() { return false; }
}
this(Widget parent = null) { this(Widget parent = null) {
super(parent); super(parent);
@ -4050,7 +4071,6 @@ version(win32_widgets) {
// import win32.winuser; // import win32.winuser;
pragma(lib, "comctl32"); pragma(lib, "comctl32");
shared static this() { shared static this() {
// http://msdn.microsoft.com/en-us/library/windows/desktop/bb775507(v=vs.85).aspx // http://msdn.microsoft.com/en-us/library/windows/desktop/bb775507(v=vs.85).aspx
INITCOMMONCONTROLSEX ic; INITCOMMONCONTROLSEX ic;
@ -4306,6 +4326,9 @@ enum GenericIcons : ushort {
Print, Print,
} }
version(win32_widgets)
pragma(lib, "comdlg32");
void getOpenFileName( void getOpenFileName(
void delegate(string) onOK = null, void delegate(string) onOK = null,
string prefilledName = null, string prefilledName = null,
@ -4398,7 +4421,7 @@ class FilePicker : Dialog {
okButton.addEventListener(EventType.triggered, &OK); okButton.addEventListener(EventType.triggered, &OK);
this.addEventListener("keydown", (Event event) { this.addEventListener("keydown", (Event event) {
if(event.key == Key.Enter) if(event.key == Key.Enter || event.key == Key.PadEnter)
OK(); OK();
if(event.key == Key.Escape) if(event.key == Key.Escape)
Cancel(); Cancel();

View File

@ -124,6 +124,11 @@ class ColorPickerDialog : Dialog {
b.content = to!string(current.b); b.content = to!string(current.b);
a.content = to!string(current.a); a.content = to!string(current.a);
r.addEventListener("focus", &r.selectAll);
g.addEventListener("focus", &g.selectAll);
b.addEventListener("focus", &b.selectAll);
a.addEventListener("focus", &a.selectAll);
if(hslImage !is null) if(hslImage !is null)
wid.addEventListener("mousedown", (Event event) { wid.addEventListener("mousedown", (Event event) {
@ -149,7 +154,7 @@ class ColorPickerDialog : Dialog {
} }
this.addEventListener("keydown", (Event event) { this.addEventListener("keydown", (Event event) {
if(event.key == Key.Enter) if(event.key == Key.Enter || event.key == Key.PadEnter)
OK(); OK();
if(event.key == Key.Escape) if(event.key == Key.Escape)
Cancel(); Cancel();
@ -185,6 +190,8 @@ class ColorPickerDialog : Dialog {
cancelButton.addEventListener(EventType.triggered, &Cancel); cancelButton.addEventListener(EventType.triggered, &Cancel);
okButton.addEventListener(EventType.triggered, &OK); okButton.addEventListener(EventType.triggered, &OK);
r.focus();
} }
LabeledLineEdit r; LabeledLineEdit r;

View File

@ -8,7 +8,7 @@
changed on the painter i guess) like font, color, size, background, changed on the painter i guess) like font, color, size, background,
etc. etc.
We can also fetch the carat location from it somehow. We can also fetch the caret location from it somehow.
Should prolly be an overload of drawText Should prolly be an overload of drawText
@ -10922,7 +10922,7 @@ mixin template ExperimentalTextComponent() {
void clear() { void clear() {
blocks = null; blocks = null;
carat = Carat.init; caret = Caret.init;
} }
void addText(Args...)(Args args) { void addText(Args...)(Args args) {
@ -10957,7 +10957,7 @@ mixin template ExperimentalTextComponent() {
ie.text = arg[lastLineIndex .. $]; ie.text = arg[lastLineIndex .. $];
ie.containingBlock = blocks[$-1]; ie.containingBlock = blocks[$-1];
blocks[$-1].parts ~= ie.clone; blocks[$-1].parts ~= ie.clone;
carat = Carat(this, blocks[$-1].parts[$-1], blocks[$-1].parts[$-1].text.length); caret = Caret(this, blocks[$-1].parts[$-1], blocks[$-1].parts[$-1].text.length);
} }
} }
} }
@ -10980,7 +10980,7 @@ mixin template ExperimentalTextComponent() {
} }
} }
// FIXME: ensure no other carats have a reference to it // FIXME: ensure no other carets have a reference to it
} }
/// Call this if the inputs change. It will reflow everything /// Call this if the inputs change. It will reflow everything
@ -11024,13 +11024,13 @@ mixin template ExperimentalTextComponent() {
return exact ? TextIdentifyResult.init : inexactMatch.fixupNewline; return exact ? TextIdentifyResult.init : inexactMatch.fixupNewline;
} }
void moveCaratToPixelCoordinates(int x, int y) { void moveCaretToPixelCoordinates(int x, int y) {
auto result = identify(x, y); auto result = identify(x, y);
carat.inlineElement = result.element; caret.inlineElement = result.element;
carat.offset = result.offset; caret.offset = result.offset;
} }
// FIXME: carat can remain sometimes when inserting // FIXME: caret can remain sometimes when inserting
// FIXME: inserting at the beginning once you already have something can eff it up. // FIXME: inserting at the beginning once you already have something can eff it up.
void drawInto(ScreenPainter painter, bool focused = false) { void drawInto(ScreenPainter painter, bool focused = false) {
//painter.setClipRectangle(boundingBox); //painter.setClipRectangle(boundingBox);
@ -11077,38 +11077,69 @@ mixin template ExperimentalTextComponent() {
} }
} }
// on every redraw, I will force the carat to be // on every redraw, I will force the caret to be
// redrawn too, in order to eliminate perceived lag // redrawn too, in order to eliminate perceived lag
// when moving around with the mouse. // when moving around with the mouse.
eraseCarat(painter); eraseCaret(painter);
if(focused) { if(focused) {
highlightSelection(painter); highlightSelection(painter);
drawCarat(painter); drawCaret(painter);
} }
} }
void highlightSelection(ScreenPainter painter) { void highlightSelection(ScreenPainter painter) {
if(selectionStart is selectionEnd)
return; // no selection
assert(selectionStart.inlineElement !is null);
assert(selectionEnd.inlineElement !is null);
painter.rasterOp = RasterOp.xor;
painter.outlineColor = Color.transparent;
painter.fillColor = Color(255, 255, 127);
auto at = selectionStart.inlineElement;
auto atOffset = selectionStart.offset;
bool done;
while(at) {
auto box = at.boundingBox;
if(atOffset < at.letterXs.length)
box.left = at.letterXs[atOffset];
if(at is selectionEnd.inlineElement) {
if(selectionEnd.offset < at.letterXs.length)
box.right = at.letterXs[selectionEnd.offset];
done = true;
}
painter.drawRectangle(box.upperLeft, box.width, box.height);
if(done)
break;
at = at.getNextInlineElement();
atOffset = 0;
}
} }
int caratLastDrawnX, caratLastDrawnY1, caratLastDrawnY2; int caretLastDrawnX, caretLastDrawnY1, caretLastDrawnY2;
bool caratShowingOnScreen = false; bool caretShowingOnScreen = false;
void drawCarat(ScreenPainter painter) { void drawCaret(ScreenPainter painter) {
//painter.setClipRectangle(boundingBox); //painter.setClipRectangle(boundingBox);
int x, y1, y2; int x, y1, y2;
if(carat.inlineElement is null) { if(caret.inlineElement is null) {
x = boundingBox.left; x = boundingBox.left;
y1 = boundingBox.top + 2; y1 = boundingBox.top + 2;
y2 = boundingBox.top + painter.fontHeight; y2 = boundingBox.top + painter.fontHeight;
} else { } else {
x = carat.inlineElement.xOfIndex(carat.offset); x = caret.inlineElement.xOfIndex(caret.offset);
y1 = carat.inlineElement.boundingBox.top + 2; y1 = caret.inlineElement.boundingBox.top + 2;
y2 = carat.inlineElement.boundingBox.bottom - 2; y2 = caret.inlineElement.boundingBox.bottom - 2;
} }
if(caratShowingOnScreen && (x != caratLastDrawnX || y1 != caratLastDrawnY1 || y2 != caratLastDrawnY2)) if(caretShowingOnScreen && (x != caretLastDrawnX || y1 != caretLastDrawnY1 || y2 != caretLastDrawnY2))
eraseCarat(painter); eraseCaret(painter);
painter.pen = Pen(Color.white, 1); painter.pen = Pen(Color.white, 1);
painter.rasterOp = RasterOp.xor; painter.rasterOp = RasterOp.xor;
@ -11117,117 +11148,181 @@ mixin template ExperimentalTextComponent() {
Point(x, y2) Point(x, y2)
); );
painter.rasterOp = RasterOp.normal; painter.rasterOp = RasterOp.normal;
caratShowingOnScreen = !caratShowingOnScreen; caretShowingOnScreen = !caretShowingOnScreen;
if(caratShowingOnScreen) { if(caretShowingOnScreen) {
caratLastDrawnX = x; caretLastDrawnX = x;
caratLastDrawnY1 = y1; caretLastDrawnY1 = y1;
caratLastDrawnY2 = y2; caretLastDrawnY2 = y2;
} }
} }
void eraseCarat(ScreenPainter painter) { void eraseCaret(ScreenPainter painter) {
//painter.setClipRectangle(boundingBox); //painter.setClipRectangle(boundingBox);
if(!caratShowingOnScreen) return; if(!caretShowingOnScreen) return;
painter.pen = Pen(Color.white, 1); painter.pen = Pen(Color.white, 1);
painter.rasterOp = RasterOp.xor; painter.rasterOp = RasterOp.xor;
painter.drawLine( painter.drawLine(
Point(caratLastDrawnX, caratLastDrawnY1), Point(caretLastDrawnX, caretLastDrawnY1),
Point(caratLastDrawnX, caratLastDrawnY2) Point(caretLastDrawnX, caretLastDrawnY2)
); );
caratShowingOnScreen = false; caretShowingOnScreen = false;
painter.rasterOp = RasterOp.normal; painter.rasterOp = RasterOp.normal;
} }
/// Carat movement api /// Caret movement api
/// These should give the user a logical result based on what they see on screen... /// These should give the user a logical result based on what they see on screen...
/// thus they locate predominately by *pixels* not char index. (These will generally coincide with monospace fonts tho!) /// thus they locate predominately by *pixels* not char index. (These will generally coincide with monospace fonts tho!)
void moveUp(ref Carat carat) { void moveUp(ref Caret caret) {
if(carat.inlineElement is null) return; if(caret.inlineElement is null) return;
auto x = carat.inlineElement.xOfIndex(carat.offset); auto x = caret.inlineElement.xOfIndex(caret.offset);
auto y = carat.inlineElement.boundingBox.top + 2; auto y = caret.inlineElement.boundingBox.top + 2;
y -= carat.inlineElement.boundingBox.bottom - carat.inlineElement.boundingBox.top; y -= caret.inlineElement.boundingBox.bottom - caret.inlineElement.boundingBox.top;
auto i = identify(x, y); auto i = identify(x, y);
if(i.element) { if(i.element) {
carat.inlineElement = i.element; caret.inlineElement = i.element;
carat.offset = i.offset; caret.offset = i.offset;
} }
} }
void moveDown(ref Carat carat) { void moveDown(ref Caret caret) {
if(carat.inlineElement is null) return; if(caret.inlineElement is null) return;
auto x = carat.inlineElement.xOfIndex(carat.offset); auto x = caret.inlineElement.xOfIndex(caret.offset);
auto y = carat.inlineElement.boundingBox.bottom - 2; auto y = caret.inlineElement.boundingBox.bottom - 2;
y += carat.inlineElement.boundingBox.bottom - carat.inlineElement.boundingBox.top; y += caret.inlineElement.boundingBox.bottom - caret.inlineElement.boundingBox.top;
auto i = identify(x, y); auto i = identify(x, y);
if(i.element) { if(i.element) {
carat.inlineElement = i.element; caret.inlineElement = i.element;
carat.offset = i.offset; caret.offset = i.offset;
} }
} }
void moveLeft(ref Carat carat) { void moveLeft(ref Caret caret) {
if(carat.inlineElement is null) return; if(caret.inlineElement is null) return;
if(carat.offset) if(caret.offset)
carat.offset--; caret.offset--;
else { else {
auto p = carat.inlineElement.getPreviousInlineElement(); auto p = caret.inlineElement.getPreviousInlineElement();
if(p) { if(p) {
carat.inlineElement = p; caret.inlineElement = p;
if(p.text.length && p.text[$-1] == '\n') if(p.text.length && p.text[$-1] == '\n')
carat.offset = p.text.length - 1; caret.offset = p.text.length - 1;
else else
carat.offset = p.text.length; caret.offset = p.text.length;
} }
} }
} }
void moveRight(ref Carat carat) { void moveRight(ref Caret caret) {
if(carat.inlineElement is null) return; if(caret.inlineElement is null) return;
if(carat.offset < carat.inlineElement.text.length && carat.inlineElement.text[carat.offset] != '\n') { if(caret.offset < caret.inlineElement.text.length && caret.inlineElement.text[caret.offset] != '\n') {
carat.offset++; caret.offset++;
} else { } else {
auto p = carat.inlineElement.getNextInlineElement(); auto p = caret.inlineElement.getNextInlineElement();
if(p) { if(p) {
carat.inlineElement = p; caret.inlineElement = p;
carat.offset = 0; caret.offset = 0;
} }
} }
} }
void moveHome(ref Carat carat) { void moveHome(ref Caret caret) {
if(carat.inlineElement is null) return; if(caret.inlineElement is null) return;
auto x = 0; auto x = 0;
auto y = carat.inlineElement.boundingBox.top + 2; auto y = caret.inlineElement.boundingBox.top + 2;
auto i = identify(x, y); auto i = identify(x, y);
if(i.element) { if(i.element) {
carat.inlineElement = i.element; caret.inlineElement = i.element;
carat.offset = i.offset; caret.offset = i.offset;
} }
} }
void moveEnd(ref Carat carat) { void moveEnd(ref Caret caret) {
if(carat.inlineElement is null) return; if(caret.inlineElement is null) return;
auto x = int.max; auto x = int.max;
auto y = carat.inlineElement.boundingBox.top + 2; auto y = caret.inlineElement.boundingBox.top + 2;
auto i = identify(x, y); auto i = identify(x, y);
if(i.element) { if(i.element) {
carat.inlineElement = i.element; caret.inlineElement = i.element;
carat.offset = i.offset; caret.offset = i.offset;
} }
} }
void movePageUp(ref Carat carat) {} void movePageUp(ref Caret caret) {}
void movePageDown(ref Carat carat) {} void movePageDown(ref Caret caret) {}
/// Plain text editing api. These work at the current carat inside the selected inline element. void moveDocumentStart(ref Caret caret) {
if(blocks.length && blocks[0].parts.length)
caret = Caret(this, blocks[0].parts[0], 0);
else
caret = Caret.init;
}
void moveDocumentEnd(ref Caret caret) {
if(blocks.length) {
auto parts = blocks[$-1].parts;
if(parts.length) {
caret = Caret(this, parts[$-1], parts[$-1].text.length);
} else {
caret = Caret.init;
}
} else
caret = Caret.init;
}
void deleteSelection() {
if(selectionStart is selectionEnd)
return;
assert(selectionStart.inlineElement !is null);
assert(selectionEnd.inlineElement !is null);
auto at = selectionStart.inlineElement;
auto atOffset = selectionStart.offset;
while(at) {
at.text = at.text[atOffset .. $];
if(at is selectionEnd.inlineElement) {
selectionEnd.offset -= atOffset;
at.text = at.text[selectionEnd.offset .. $];
selectionEnd.offset = 0;
break;
} else {
auto cfd = at;
at = at.getNextInlineElement();
if(at)
atOffset = at.text.length;
if(cfd.text.length == 0) {
// and remove cfd
for(size_t a = 0; a < cfd.containingBlock.parts.length; a++) {
if(cfd.containingBlock.parts[a] is cfd) {
for(size_t i = a; i < cfd.containingBlock.parts.length - 1; i++)
cfd.containingBlock.parts[i] = cfd.containingBlock.parts[i + 1];
cfd.containingBlock.parts = cfd.containingBlock.parts[0 .. $-1];
}
}
}
}
}
caret = selectionEnd;
selectNone();
}
/// Plain text editing api. These work at the current caret inside the selected inline element.
void insert(string text) {} void insert(string text) {}
void insert(dchar ch) { void insert(dchar ch) {
deleteSelection();
if(ch == 127) { if(ch == 127) {
delete_(); delete_();
return; return;
@ -11237,32 +11332,32 @@ mixin template ExperimentalTextComponent() {
return; return;
} }
if(ch == 13) ch = 10; if(ch == 13) ch = 10;
auto e = carat.inlineElement; auto e = caret.inlineElement;
if(e is null) { if(e is null) {
addText("" ~ cast(char) ch) ; // FIXME addText("" ~ cast(char) ch) ; // FIXME
return; return;
} }
if(carat.offset == e.text.length) { if(caret.offset == e.text.length) {
e.text ~= cast(char) ch; // FIXME e.text ~= cast(char) ch; // FIXME
carat.offset++; caret.offset++;
if(ch == 10) { if(ch == 10) {
auto c = carat.inlineElement.clone; auto c = caret.inlineElement.clone;
c.text = null; c.text = null;
insertPartAfter(c,e); insertPartAfter(c,e);
carat = Carat(this, c, 0); caret = Caret(this, c, 0);
} }
} else { } else {
// FIXME cast char sucks // FIXME cast char sucks
if(ch == 10) { if(ch == 10) {
auto c = carat.inlineElement.clone; auto c = caret.inlineElement.clone;
c.text = e.text[carat.offset .. $]; c.text = e.text[caret.offset .. $];
e.text = e.text[0 .. carat.offset] ~ cast(char) ch; e.text = e.text[0 .. caret.offset] ~ cast(char) ch;
insertPartAfter(c,e); insertPartAfter(c,e);
carat = Carat(this, c, 0); caret = Caret(this, c, 0);
} else { } else {
e.text = e.text[0 .. carat.offset] ~ cast(char) ch ~ e.text[carat.offset .. $]; e.text = e.text[0 .. caret.offset] ~ cast(char) ch ~ e.text[caret.offset .. $];
carat.offset++; caret.offset++;
} }
} }
} }
@ -11300,48 +11395,53 @@ mixin template ExperimentalTextComponent() {
void backspace() { void backspace() {
try_again: try_again:
auto e = carat.inlineElement; auto e = caret.inlineElement;
if(e is null) if(e is null)
return; return;
if(carat.offset == 0) { if(caret.offset == 0) {
auto prev = e.getPreviousInlineElement(); auto prev = e.getPreviousInlineElement();
if(prev is null) if(prev is null)
return; return;
auto newOffset = prev.text.length; auto newOffset = prev.text.length;
tryMerge(prev, e); tryMerge(prev, e);
carat.inlineElement = prev; caret.inlineElement = prev;
carat.offset = prev is null ? 0 : newOffset; caret.offset = prev is null ? 0 : newOffset;
goto try_again; goto try_again;
} else if(carat.offset == e.text.length) { } else if(caret.offset == e.text.length) {
e.text = e.text[0 .. $-1]; e.text = e.text[0 .. $-1];
carat.offset--; caret.offset--;
} else { } else {
e.text = e.text[0 .. carat.offset - 1] ~ e.text[carat.offset .. $]; e.text = e.text[0 .. caret.offset - 1] ~ e.text[caret.offset .. $];
carat.offset--; caret.offset--;
} }
//cleanupStructures(); //cleanupStructures();
} }
void delete_() { void delete_() {
auto after = carat; auto after = caret;
moveRight(after); moveRight(after);
if(carat != after) { if(caret != after) {
carat = after; caret = after;
backspace(); backspace();
} }
} }
void overstrike() {} void overstrike() {}
/// Selection API. See also: carat movement. /// Selection API. See also: caret movement.
void selectAll() {} void selectAll() {
void selectNone() {} moveDocumentStart(selectionStart);
moveDocumentEnd(selectionEnd);
}
void selectNone() {
selectionStart = selectionEnd = Caret.init;
}
/// Rich text editing api. These allow you to manipulate the meta data of the current element and add new elements. /// Rich text editing api. These allow you to manipulate the meta data of the current element and add new elements.
/// They will modify the current selection if there is one and will splice one in if needed. /// They will modify the current selection if there is one and will splice one in if needed.
void changeAttributes() {} void changeAttributes() {}
/// Text search api. They manipulate the selection and/or carat. /// Text search api. They manipulate the selection and/or caret.
void findText(string text) {} void findText(string text) {}
void findIndex(size_t textIndex) {} void findIndex(size_t textIndex) {}
@ -11362,17 +11462,17 @@ mixin template ExperimentalTextComponent() {
} }
bool contentEditable; // can it be edited? bool contentEditable; // can it be edited?
bool contentCaratable; // is there a carat/cursor that moves around in there? bool contentCaretable; // is there a caret/cursor that moves around in there?
bool contentSelectable; // selectable? bool contentSelectable; // selectable?
Carat carat; Caret caret;
Carat selectionStart; Caret selectionStart;
Carat selectionEnd; Caret selectionEnd;
bool insertMode; bool insertMode;
} }
struct Carat { struct Caret {
TextLayout layout; TextLayout layout;
InlineElement inlineElement; InlineElement inlineElement;
size_t offset; size_t offset;