using System; using System.Collections.Generic; using Unity.UIWidgets.animation; using Unity.UIWidgets.async; using Unity.UIWidgets.foundation; using Unity.UIWidgets.painting; using Unity.UIWidgets.rendering; using Unity.UIWidgets.scheduler; using Unity.UIWidgets.service; using Unity.UIWidgets.ui; using UnityEngine; using Color = Unity.UIWidgets.ui.Color; using Rect = Unity.UIWidgets.ui.Rect; using TextStyle = Unity.UIWidgets.painting.TextStyle; namespace Unity.UIWidgets.widgets { public delegate void SelectionChangedCallback(TextSelection selection, SelectionChangedCause cause); public class TextEditingController : ValueNotifier { public TextEditingController(string text) : base(text == null ? TextEditingValue.empty : new TextEditingValue(text)) { } TextEditingController(TextEditingValue value) : base(value ?? TextEditingValue.empty) { } public TextEditingController fromValue(TextEditingValue value) { return new TextEditingController(value); } public string text { get { return this.value.text; } set { this.value = this.value.copyWith(text: value, selection: TextSelection.collapsed(-1), composing: TextRange.empty); } } public TextSelection selection { get { return this.value.selection; } set { if (value.start > this.text.Length || value.end > this.text.Length) { throw new UIWidgetsError($"invalid text selection: {value}"); } this.value = this.value.copyWith(selection: value, composing: TextRange.empty); } } public void clear() { this.value = TextEditingValue.empty; } public void clearComposing() { this.value = this.value.copyWith(composing: TextRange.empty); } } public class EditableText : StatefulWidget { public readonly TextEditingController controller; public readonly FocusNode focusNode; public readonly bool obscureText; public readonly bool autocorrect; public readonly TextStyle style; public readonly TextAlign textAlign; public readonly TextDirection? textDirection; public readonly TextCapitalization textCapitalization; public readonly float? textScaleFactor; public readonly Color cursorColor; public readonly int? maxLines; public readonly bool autofocus; public readonly Color selectionColor; public readonly TextSelectionControls selectionControls; public readonly TextInputType keyboardType; public readonly TextInputAction? textInputAction; public readonly ValueChanged onChanged; public readonly VoidCallback onEditingComplete; public readonly ValueChanged onSubmitted; public readonly SelectionChangedCallback onSelectionChanged; public readonly List inputFormatters; public readonly bool rendererIgnoresPointer; public readonly bool unityTouchKeyboard; public EditableText(TextEditingController controller, FocusNode focusNode, TextStyle style, Color cursorColor, bool obscureText = false, bool autocorrect = false, TextAlign textAlign = TextAlign.left, TextDirection? textDirection = null, float? textScaleFactor = null, int? maxLines = 1, bool autofocus = false, Color selectionColor = null, TextSelectionControls selectionControls = null, TextInputType keyboardType = null, TextInputAction? textInputAction = null, TextCapitalization textCapitalization = TextCapitalization.none, ValueChanged onChanged = null, VoidCallback onEditingComplete = null, ValueChanged onSubmitted = null, SelectionChangedCallback onSelectionChanged = null, List inputFormatters = null, bool rendererIgnoresPointer = false, EdgeInsets scrollPadding = null, bool unityTouchKeyboard = false, Key key = null) : base(key) { D.assert(controller != null); D.assert(focusNode != null); D.assert(style != null); D.assert(cursorColor != null); D.assert(maxLines == null || maxLines > 0); this.keyboardType = keyboardType ?? (maxLines == 1 ? TextInputType.text : TextInputType.multiline); this.scrollPadding = scrollPadding ?? EdgeInsets.all(20.0f); this.controller = controller; this.focusNode = focusNode; this.obscureText = obscureText; this.autocorrect = autocorrect; this.style = style; this.textAlign = textAlign; this.textDirection = textDirection; this.textScaleFactor = textScaleFactor; this.textInputAction = textInputAction; this.textCapitalization = textCapitalization; this.cursorColor = cursorColor; this.maxLines = maxLines; this.autofocus = autofocus; this.selectionColor = selectionColor; this.onChanged = onChanged; this.onSubmitted = onSubmitted; this.onSelectionChanged = onSelectionChanged; this.onEditingComplete = onEditingComplete; this.rendererIgnoresPointer = rendererIgnoresPointer; this.selectionControls = selectionControls; this.unityTouchKeyboard = unityTouchKeyboard; if (maxLines == 1) { this.inputFormatters = new List(); this.inputFormatters.Add(BlacklistingTextInputFormatter.singleLineFormatter); if (inputFormatters != null) { this.inputFormatters.AddRange(inputFormatters); } } else { this.inputFormatters = inputFormatters; } } public readonly EdgeInsets scrollPadding; public override State createState() { return new EditableTextState(); } public override void debugFillProperties(DiagnosticPropertiesBuilder properties) { base.debugFillProperties(properties); properties.add(new DiagnosticsProperty("controller", this.controller)); properties.add(new DiagnosticsProperty("focusNode", this.focusNode)); properties.add(new DiagnosticsProperty("obscureText", this.obscureText, defaultValue: false)); properties.add(new DiagnosticsProperty("autocorrect", this.autocorrect, defaultValue: true)); if (this.style != null) { this.style.debugFillProperties(properties); } properties.add(new EnumProperty("textAlign", this.textAlign, defaultValue: Diagnostics.kNullDefaultValue)); properties.add(new EnumProperty("textDirection", this.textDirection, defaultValue: Diagnostics.kNullDefaultValue)); properties.add(new DiagnosticsProperty("textScaleFactor", this.textScaleFactor, defaultValue: Diagnostics.kNullDefaultValue)); properties.add(new DiagnosticsProperty("maxLines", this.maxLines, defaultValue: 1)); properties.add(new DiagnosticsProperty("autofocus", this.autofocus, defaultValue: false)); properties.add( new DiagnosticsProperty("keyboardType", this.keyboardType, defaultValue: null)); } } public class EditableTextState : AutomaticKeepAliveClientMixin, TextInputClient, TextSelectionDelegate { const int _kObscureShowLatestCharCursorTicks = 3; static TimeSpan _kCursorBlinkHalfPeriod = TimeSpan.FromMilliseconds(500); Timer _cursorTimer; ValueNotifier _showCursor = new ValueNotifier(false); GlobalKey _editableKey = GlobalKey.key(); LayerLink _layerLink = new LayerLink(); bool _didAutoFocus = false; public ScrollController _scrollController = new ScrollController(); TextInputConnection _textInputConnection; TextSelectionOverlay _selectionOverlay; int _obscureShowCharTicksPending = 0; int _obscureLatestCharIndex; bool _textChangedSinceLastCaretUpdate = false; protected override bool wantKeepAlive { get { return this.widget.focusNode.hasFocus; } } public override void initState() { base.initState(); this.widget.controller.addListener(this._didChangeTextEditingValue); this.widget.focusNode.addListener(this._handleFocusChanged); this._scrollController.addListener(() => { this._selectionOverlay?.updateForScroll(); }); } public override void didChangeDependencies() { base.didChangeDependencies(); if (!this._didAutoFocus && this.widget.autofocus) { FocusScope.of(this.context).autofocus(this.widget.focusNode); this._didAutoFocus = true; } } public override void didUpdateWidget(StatefulWidget old) { EditableText oldWidget = (EditableText) old; base.didUpdateWidget(oldWidget); if (this.widget.controller != oldWidget.controller) { oldWidget.controller.removeListener(this._didChangeTextEditingValue); this.widget.controller.addListener(this._didChangeTextEditingValue); this._updateRemoteEditingValueIfNeeded(); this._updateImePosIfNeed(); } if (this.widget.focusNode != oldWidget.focusNode) { oldWidget.focusNode.removeListener(this._handleFocusChanged); this.widget.focusNode.addListener(this._handleFocusChanged); this.updateKeepAlive(); } } public override void dispose() { this.widget.controller.removeListener(this._didChangeTextEditingValue); this._closeInputConnectionIfNeeded(); D.assert(!this._hasInputConnection); this._stopCursorTimer(); D.assert(this._cursorTimer == null); this._selectionOverlay?.dispose(); this._selectionOverlay = null; this.widget.focusNode.removeListener(this._handleFocusChanged); base.dispose(); } TextEditingValue _lastKnownRemoteTextEditingValue; public void updateEditingValue(TextEditingValue value) { if (value.text != this._value.text) { this._hideSelectionOverlayIfNeeded(); this._showCaretOnScreen(); if (this.widget.obscureText && value.text.Length == this._value.text.Length + 1) { this._obscureShowCharTicksPending = !this._unityKeyboard() ? _kObscureShowLatestCharCursorTicks : 0; this._obscureLatestCharIndex = this._value.selection.baseOffset; } } this._lastKnownRemoteTextEditingValue = value; this._formatAndSetValue(value); } public TextEditingValue getValueForAction(TextInputAction operation) { TextPosition newPosition = null; TextPosition newExtend = null; TextEditingValue newValue = null; TextSelection newSelection = null; TextPosition startPos = new TextPosition(this._value.selection.start, this._value.selection.affinity); switch (operation) { case TextInputAction.moveLeft: newValue = this._value.moveLeft(); break; case TextInputAction.moveRight: newValue = this._value.moveRight(); break; case TextInputAction.moveUp: newPosition = this.renderEditable.getPositionUp(startPos); break; case TextInputAction.moveDown: newPosition = this.renderEditable.getPositionDown(startPos); break; case TextInputAction.moveLineStart: newPosition = this.renderEditable.getParagraphStart(startPos, TextAffinity.downstream); break; case TextInputAction.moveLineEnd: newPosition = this.renderEditable.getParagraphEnd(startPos, TextAffinity.upstream); break; case TextInputAction.moveWordRight: newPosition = this.renderEditable.getWordRight(startPos); break; case TextInputAction.moveWordLeft: newPosition = this.renderEditable.getWordLeft(startPos); break; // case TextInputAction.MoveToStartOfNextWord: MoveToStartOfNextWord(); break; // case TextInputAction.MoveToEndOfPreviousWord: MoveToEndOfPreviousWord(); break; case TextInputAction.moveTextStart: newPosition = new TextPosition(0); break; case TextInputAction.moveTextEnd: newPosition = new TextPosition(this._value.text.Length); break; case TextInputAction.moveParagraphForward: newPosition = this.renderEditable.getParagraphForward(startPos); break; case TextInputAction.moveParagraphBackward: newPosition = this.renderEditable.getParagraphBackward(startPos); break; case TextInputAction.moveGraphicalLineStart: newPosition = this.renderEditable.getLineStartPosition(startPos, TextAffinity.downstream); break; case TextInputAction.moveGraphicalLineEnd: newPosition = this.renderEditable.getLineEndPosition(startPos, TextAffinity.upstream); break; case TextInputAction.selectLeft: newValue = this._value.extendLeft(); break; case TextInputAction.selectRight: newValue = this._value.extendRight(); break; case TextInputAction.selectUp: newExtend = this.renderEditable.getPositionUp(this._value.selection.extendPos); break; case TextInputAction.selectDown: newExtend = this.renderEditable.getPositionDown(this._value.selection.extendPos); break; case TextInputAction.selectWordRight: newExtend = this.renderEditable.getWordRight(this._value.selection.extendPos); break; case TextInputAction.selectWordLeft: newExtend = this.renderEditable.getWordLeft(this._value.selection.extendPos); break; // case TextInputAction.SelectToEndOfPreviousWord: SelectToEndOfPreviousWord(); break; // case TextInputAction.SelectToStartOfNextWord: SelectToStartOfNextWord(); break; // case TextInputAction.selectTextStart: newExtend = new TextPosition(0); break; case TextInputAction.selectTextEnd: newExtend = new TextPosition(this._value.text.Length); break; case TextInputAction.expandSelectGraphicalLineStart: if (this._value.selection.isCollapsed || !this.renderEditable.isLineEndOrStart(this._value.selection.start)) { newSelection = new TextSelection(this.renderEditable.getLineStartPosition(startPos).offset, this._value.selection.end, this._value.selection.affinity); } break; case TextInputAction.expandSelectGraphicalLineEnd: if (this._value.selection.isCollapsed || !this.renderEditable.isLineEndOrStart(this._value.selection.end)) { newSelection = new TextSelection(this._value.selection.start, this.renderEditable.getLineEndPosition(this._value.selection.endPos).offset, this._value.selection.affinity); } break; case TextInputAction.selectParagraphForward: newExtend = this.renderEditable.getParagraphForward(this._value.selection.extendPos); break; case TextInputAction.selectParagraphBackward: newExtend = this.renderEditable.getParagraphBackward(this._value.selection.extendPos); break; case TextInputAction.selectGraphicalLineStart: newExtend = this.renderEditable.getLineStartPosition(this._value.selection.extendPos); break; case TextInputAction.selectGraphicalLineEnd: newExtend = this.renderEditable.getLineEndPosition(startPos); break; case TextInputAction.delete: newValue = this._value.deleteSelection(false); break; case TextInputAction.backspace: newValue = this._value.deleteSelection(); break; case TextInputAction.selectAll: newSelection = this._value.selection.copyWith(baseOffset: 0, extentOffset: this._value.text.Length); break; } if (newPosition != null) { return this._value.copyWith(selection: TextSelection.fromPosition(newPosition)); } else if (newExtend != null) { return this._value.copyWith(selection: this._value.selection.copyWith(extentOffset: newExtend.offset)); } else if (newSelection != null) { return this._value.copyWith(selection: newSelection); } else if (newValue != null) { return newValue; } return this._value; } public void performAction(TextInputAction action) { TextEditingValue newValue; switch (action) { case TextInputAction.newline: if (this.widget.maxLines == 1) { this._finalizeEditing(true); } break; case TextInputAction.done: case TextInputAction.go: case TextInputAction.send: case TextInputAction.search: this._finalizeEditing(true); break; case TextInputAction.next: case TextInputAction.previous: case TextInputAction.continueAction: case TextInputAction.join: case TextInputAction.route: case TextInputAction.emergencyCall: this._finalizeEditing(false); break; default: newValue = this.getValueForAction(action); if (newValue != this.textEditingValue) { this.textEditingValue = newValue; } break; } } void _finalizeEditing(bool shouldUnfocus) { if (this.widget.onEditingComplete != null) { this.widget.onEditingComplete(); } else { this.widget.controller.clearComposing(); if (shouldUnfocus) { this.widget.focusNode.unfocus(); } } if (this.widget.onSubmitted != null) { this.widget.onSubmitted(this._value.text); } } void _updateRemoteEditingValueIfNeeded() { if (!this._hasInputConnection) { return; } var localValue = this._value; if (localValue == this._lastKnownRemoteTextEditingValue) { return; } this._lastKnownRemoteTextEditingValue = localValue; this._textInputConnection.setEditingState(localValue); } // Calculate the new scroll offset so the cursor remains visible. float _getScrollOffsetForCaret(Rect caretRect) { float caretStart = this._isMultiline ? caretRect.top : caretRect.left; float caretEnd = this._isMultiline ? caretRect.bottom : caretRect.right; float scrollOffset = this._scrollController.offset; float viewportExtent = this._scrollController.position.viewportDimension; if (caretStart < 0.0) { scrollOffset += caretStart; } else if (caretEnd >= viewportExtent) { scrollOffset += caretEnd - viewportExtent; } return scrollOffset; } // Calculates where the `caretRect` would be if `_scrollController.offset` is set to `scrollOffset`. Rect _getCaretRectAtScrollOffset(Rect caretRect, float scrollOffset) { float offsetDiff = this._scrollController.offset - scrollOffset; return this._isMultiline ? caretRect.translate(0.0f, offsetDiff) : caretRect.translate(offsetDiff, 0.0f); } bool _hasInputConnection { get { return this._textInputConnection != null && this._textInputConnection.attached; } } void _openInputConnection() { if (!this._hasInputConnection) { TextEditingValue localValue = this._value; this._lastKnownRemoteTextEditingValue = localValue; this._textInputConnection = TextInput.attach(this, new TextInputConfiguration( inputType: this.widget.keyboardType, obscureText: this.widget.obscureText, autocorrect: this.widget.autocorrect, inputAction: this.widget.textInputAction ?? ((this.widget.keyboardType == TextInputType.multiline) ? TextInputAction.newline : TextInputAction.done), textCapitalization: this.widget.textCapitalization, unityTouchKeyboard: this.widget.unityTouchKeyboard )); this._textInputConnection.setEditingState(localValue); this._updateImePosIfNeed(); } this._textInputConnection.show(); } void _closeInputConnectionIfNeeded() { if (this._hasInputConnection) { this._textInputConnection.close(); this._textInputConnection = null; this._lastKnownRemoteTextEditingValue = null; } } void _openOrCloseInputConnectionIfNeeded() { if (this._hasFocus && this.widget.focusNode.consumeKeyboardToken()) { this._openInputConnection(); } else if (!this._hasFocus) { this._closeInputConnectionIfNeeded(); this.widget.controller.clearComposing(); } } public void requestKeyboard() { if (this._hasFocus) { this._openInputConnection(); } else { FocusScope.of(this.context).requestFocus(this.widget.focusNode); } } void _hideSelectionOverlayIfNeeded() { this._selectionOverlay?.hide(); this._selectionOverlay = null; } void _updateOrDisposeSelectionOverlayIfNeeded() { if (this._selectionOverlay != null) { if (this._hasFocus) { this._selectionOverlay.update(this._value); } else { this._selectionOverlay.dispose(); this._selectionOverlay = null; } } } void _handleSelectionChanged(TextSelection selection, RenderEditable renderObject, SelectionChangedCause cause) { this.widget.controller.selection = selection; this.requestKeyboard(); this._hideSelectionOverlayIfNeeded(); if (this.widget.selectionControls != null && Application.isMobilePlatform && !this._unityKeyboard()) { this._selectionOverlay = new TextSelectionOverlay( context: this.context, value: this._value, debugRequiredFor: this.widget, layerLink: this._layerLink, renderObject: renderObject, selectionControls: this.widget.selectionControls, selectionDelegate: this ); bool longPress = cause == SelectionChangedCause.longPress; if (cause != SelectionChangedCause.keyboard && (this._value.text.isNotEmpty() || longPress)) { this._selectionOverlay.showHandles(); } if (longPress || cause == SelectionChangedCause.doubleTap) { this._selectionOverlay.showToolbar(); } } if (this.widget.onSelectionChanged != null) { this.widget.onSelectionChanged(selection, cause); } } Rect _currentCaretRect; void _handleCaretChanged(Rect caretRect) { this._currentCaretRect = caretRect; // If the caret location has changed due to an update to the text or // selection, then scroll the caret into view. if (this._textChangedSinceLastCaretUpdate) { this._textChangedSinceLastCaretUpdate = false; this._showCaretOnScreen(); } } // Animation configuration for scrolling the caret back on screen. static readonly TimeSpan _caretAnimationDuration = TimeSpan.FromMilliseconds(100); static readonly Curve _caretAnimationCurve = Curves.fastOutSlowIn; bool _showCaretOnScreenScheduled = false; void _showCaretOnScreen() { if (this._showCaretOnScreenScheduled) { return; } this._showCaretOnScreenScheduled = true; SchedulerBinding.instance.addPostFrameCallback(_ => { this._showCaretOnScreenScheduled = false; if (this._currentCaretRect == null || !this._scrollController.hasClients) { return; } float scrollOffsetForCaret = this._getScrollOffsetForCaret(this._currentCaretRect); this._scrollController.animateTo(scrollOffsetForCaret, duration: _caretAnimationDuration, curve: _caretAnimationCurve); Rect newCaretRect = this._getCaretRectAtScrollOffset(this._currentCaretRect, scrollOffsetForCaret); // Enlarge newCaretRect by scrollPadding to ensure that caret is not positioned directly at the edge after scrolling. Rect inflatedRect = Rect.fromLTRB( newCaretRect.left - this.widget.scrollPadding.left, newCaretRect.top - this.widget.scrollPadding.top, newCaretRect.right + this.widget.scrollPadding.right, newCaretRect.bottom + this.widget.scrollPadding.bottom ); this._editableKey.currentContext.findRenderObject().showOnScreen( rect: inflatedRect, duration: _caretAnimationDuration, curve: _caretAnimationCurve ); }); } void _formatAndSetValue(TextEditingValue value) { var textChanged = (this._value == null ? null : this._value.text) != (value == null ? null : value.text); if (this.widget.inputFormatters != null && this.widget.inputFormatters.isNotEmpty()) { foreach (var formatter in this.widget.inputFormatters) { value = formatter.formatEditUpdate(this._value, value); } this._value = value; this._updateRemoteEditingValueIfNeeded(); this._updateImePosIfNeed(); } else { this._value = value; } if (textChanged && this.widget.onChanged != null) { this.widget.onChanged(value.text); } } public bool cursorCurrentlyVisible { get { return this._showCursor.value; } } public TimeSpan cursorBlinkInterval { get { return _kCursorBlinkHalfPeriod; } } void _cursorTick() { this._showCursor.value = !this._unityKeyboard() && !this._showCursor.value; if (this._obscureShowCharTicksPending > 0) { this.setState(() => { this._obscureShowCharTicksPending--; }); } } void _startCursorTimer() { this._showCursor.value = !this._unityKeyboard(); this._cursorTimer = Window.instance.run(_kCursorBlinkHalfPeriod, this._cursorTick, periodic: true); } void _stopCursorTimer() { if (this._cursorTimer != null) { this._cursorTimer.cancel(); } this._cursorTimer = null; this._showCursor.value = false; this._obscureShowCharTicksPending = 0; } void _startOrStopCursorTimerIfNeeded() { if (this._cursorTimer == null && this._hasFocus && this._value.selection.isCollapsed) { this._startCursorTimer(); } else if (this._cursorTimer != null && (!this._hasFocus || !this._value.selection.isCollapsed)) { this._stopCursorTimer(); } } TextEditingValue _value { get { return this.widget.controller.value; } set { this.widget.controller.value = value; } } bool _hasFocus { get { return this.widget.focusNode.hasFocus; } } bool _isMultiline { get { return this.widget.maxLines != 1; } } void _didChangeTextEditingValue() { this._updateRemoteEditingValueIfNeeded(); this._updateImePosIfNeed(); this._startOrStopCursorTimerIfNeeded(); this._updateOrDisposeSelectionOverlayIfNeeded(); this._textChangedSinceLastCaretUpdate = true; this.setState(() => { }); } void _handleFocusChanged() { this._openOrCloseInputConnectionIfNeeded(); this._startOrStopCursorTimerIfNeeded(); this._updateOrDisposeSelectionOverlayIfNeeded(); if (!this._hasFocus) { this._value = new TextEditingValue(text: this._value.text); } else if (!this._value.selection.isValid) { this.widget.controller.selection = TextSelection.collapsed(offset: this._value.text.Length); } this.updateKeepAlive(); } TextDirection? _textDirection { get { TextDirection? result = this.widget.textDirection ?? Directionality.of(this.context); D.assert(result != null, $"{this.GetType().FullName} created without a textDirection and with no ambient Directionality."); return result; } } public RenderEditable renderEditable { get { return (RenderEditable) this._editableKey.currentContext.findRenderObject(); } } public TextEditingValue textEditingValue { get { return this._value; } set { this._selectionOverlay?.update(value); this._formatAndSetValue(value); } } public void bringIntoView(TextPosition position) { this._scrollController.jumpTo( this._getScrollOffsetForCaret(this.renderEditable.getLocalRectForCaret(position))); } public void hideToolbar() { this._selectionOverlay?.hide(); } public override Widget build(BuildContext context) { FocusScope.of(context).reparentIfNeeded(this.widget.focusNode); base.build(context); // See AutomaticKeepAliveClientMixin. return new Scrollable( axisDirection: this._isMultiline ? AxisDirection.down : AxisDirection.right, controller: this._scrollController, physics: new ClampingScrollPhysics(), viewportBuilder: (BuildContext _context, ViewportOffset offset) => new CompositedTransformTarget( link: this._layerLink, child: new _Editable( key: this._editableKey, textSpan: this.buildTextSpan(), value: this._value, cursorColor: this.widget.cursorColor, showCursor: this._showCursor, hasFocus: this._hasFocus, maxLines: this.widget.maxLines, selectionColor: this.widget.selectionColor, textScaleFactor: this.widget.textScaleFactor ?? MediaQuery.textScaleFactorOf(context), textAlign: this.widget.textAlign, textDirection: this._textDirection, obscureText: this.widget.obscureText, autocorrect: this.widget.autocorrect, offset: offset, onSelectionChanged: this._handleSelectionChanged, onCaretChanged: this._handleCaretChanged, rendererIgnoresPointer: this.widget.rendererIgnoresPointer ) ) ); } public TextSpan buildTextSpan() { if (!this.widget.obscureText && this._value.composing.isValid) { TextStyle composingStyle = this.widget.style.merge( new TextStyle(decoration: TextDecoration.underline) ); return new TextSpan( style: this.widget.style, children: new List { new TextSpan(text: this._value.composing.textBefore(this._value.text)), new TextSpan( style: composingStyle, text: this._value.composing.textInside(this._value.text) ), new TextSpan(text: this._value.composing.textAfter(this._value.text)), }); } var text = this._value.text; if (this.widget.obscureText) { text = new string(RenderEditable.obscuringCharacter, text.Length); int o = this._obscureShowCharTicksPending > 0 ? this._obscureLatestCharIndex : -1; if (o >= 0 && o < text.Length) { text = text.Substring(0, o) + this._value.text.Substring(o, 1) + text.Substring(o + 1); } } return new TextSpan(style: this.widget.style, text: text); } // unity keyboard has a preview view with editing function, text selection & cursor function of this widget is disable // in the case bool _unityKeyboard() { return TouchScreenKeyboard.isSupported && this.widget.unityTouchKeyboard; } Offset _getImePos() { if (this._hasInputConnection && this._textInputConnection.imeRequired()) { var localPos = this.renderEditable.getLocalRectForCaret(this._value.selection.basePos).bottomLeft; return this.renderEditable.localToGlobal(localPos); } return null; } bool _imePosUpdateScheduled = false; void _updateImePosIfNeed() { if (!this._hasInputConnection || !this._textInputConnection.imeRequired()) { return; } if (this._imePosUpdateScheduled) { return; } this._imePosUpdateScheduled = true; SchedulerBinding.instance.addPostFrameCallback(_ => { this._imePosUpdateScheduled = false; if (!this._hasInputConnection) { return; } this._textInputConnection.setIMEPos(this._getImePos()); }); } } class _Editable : LeafRenderObjectWidget { public readonly TextSpan textSpan; public readonly TextEditingValue value; public readonly Color cursorColor; public readonly ValueNotifier showCursor; public readonly bool hasFocus; public readonly int? maxLines; public readonly Color selectionColor; public readonly float textScaleFactor; public readonly TextAlign textAlign; public readonly TextDirection? textDirection; public readonly bool obscureText; public readonly bool autocorrect; public readonly ViewportOffset offset; public readonly SelectionChangedHandler onSelectionChanged; public readonly CaretChangedHandler onCaretChanged; public readonly bool rendererIgnoresPointer; public _Editable(TextSpan textSpan = null, TextEditingValue value = null, Color cursorColor = null, ValueNotifier showCursor = null, bool hasFocus = false, int? maxLines = null, Color selectionColor = null, float textScaleFactor = 1.0f, TextDirection? textDirection = null, bool obscureText = false, TextAlign textAlign = TextAlign.left, bool autocorrect = false, ViewportOffset offset = null, SelectionChangedHandler onSelectionChanged = null, CaretChangedHandler onCaretChanged = null, bool rendererIgnoresPointer = false, Key key = null) : base(key) { this.textSpan = textSpan; this.value = value; this.cursorColor = cursorColor; this.showCursor = showCursor; this.hasFocus = hasFocus; this.maxLines = maxLines; this.selectionColor = selectionColor; this.textScaleFactor = textScaleFactor; this.textAlign = textAlign; this.textDirection = textDirection; this.obscureText = obscureText; this.autocorrect = autocorrect; this.offset = offset; this.onSelectionChanged = onSelectionChanged; this.onCaretChanged = onCaretChanged; this.rendererIgnoresPointer = rendererIgnoresPointer; } public override RenderObject createRenderObject(BuildContext context) { return new RenderEditable( text: this.textSpan, textDirection: this.textDirection ?? TextDirection.ltr, offset: this.offset, showCursor: this.showCursor, cursorColor: this.cursorColor, hasFocus: this.hasFocus, maxLines: this.maxLines, selectionColor: this.selectionColor, textScaleFactor: this.textScaleFactor, textAlign: this.textAlign, selection: this.value.selection, obscureText: this.obscureText, onSelectionChanged: this.onSelectionChanged, onCaretChanged: this.onCaretChanged, ignorePointer: this.rendererIgnoresPointer ); } public override void updateRenderObject(BuildContext context, RenderObject renderObject) { var edit = (RenderEditable) renderObject; edit.text = this.textSpan; edit.cursorColor = this.cursorColor; edit.showCursor = this.showCursor; edit.hasFocus = this.hasFocus; edit.maxLines = this.maxLines; edit.selectionColor = this.selectionColor; edit.textScaleFactor = this.textScaleFactor; edit.textAlign = this.textAlign; edit.textDirection = this.textDirection; edit.selection = this.value.selection; edit.offset = this.offset; edit.onSelectionChanged = this.onSelectionChanged; edit.onCaretChanged = this.onCaretChanged; edit.ignorePointer = this.rendererIgnoresPointer; edit.obscureText = this.obscureText; } } }