using System; using System.Collections.Generic; using Unity.UIWidgets.external.simplejson; using Unity.UIWidgets.foundation; using Unity.UIWidgets.ui; using UnityEngine; namespace Unity.UIWidgets.service { public enum FloatingCursorDragState { Start, Update, End } public class RawFloatingCursorPoint { public RawFloatingCursorPoint( Offset offset = null, FloatingCursorDragState? state = null ) { D.assert(state != null); D.assert(state == FloatingCursorDragState.Update ? offset != null : true); this.offset = offset; this.state = state; } public readonly Offset offset; public readonly FloatingCursorDragState? state; } public class TextInputType : IEquatable { public readonly int index; public readonly bool? signed; public readonly bool? decimalNum; TextInputType(int index, bool? signed = null, bool? decimalNum = null) { this.index = index; this.signed = signed; this.decimalNum = decimalNum; } public static TextInputType numberWithOptions(bool signed = false, bool decimalNum = false) { return new TextInputType(2, signed: signed, decimalNum: decimalNum); } public static readonly TextInputType text = new TextInputType(0); public static readonly TextInputType multiline = new TextInputType(1); public static readonly TextInputType number = numberWithOptions(); public static readonly TextInputType phone = new TextInputType(3); public static readonly TextInputType datetime = new TextInputType(4); public static readonly TextInputType emailAddress = new TextInputType(5); public static readonly TextInputType url = new TextInputType(6); public static List _names = new List { "text", "multiline", "number", "phone", "datetime", "emailAddress", "url" }; public JSONNode toJson() { JSONObject jsonObject = new JSONObject(); jsonObject["name"] = _name; jsonObject["signed"] = signed; jsonObject["decimal"] = decimalNum; return jsonObject; } string _name { get { return $"TextInputType.{_names[index]}"; } } public bool Equals(TextInputType other) { if (ReferenceEquals(null, other)) { return false; } if (ReferenceEquals(this, other)) { return true; } return index == other.index && signed == other.signed && decimalNum == other.decimalNum; } public override bool Equals(object obj) { if (ReferenceEquals(null, obj)) { return false; } if (ReferenceEquals(this, obj)) { return true; } if (obj.GetType() != GetType()) { return false; } return Equals((TextInputType) obj); } public override int GetHashCode() { unchecked { var hashCode = index; hashCode = (hashCode * 397) ^ signed.GetHashCode(); hashCode = (hashCode * 397) ^ decimalNum.GetHashCode(); return hashCode; } } public static bool operator ==(TextInputType left, TextInputType right) { return Equals(left, right); } public static bool operator !=(TextInputType left, TextInputType right) { return !Equals(left, right); } public override string ToString() { return $"{GetType().FullName}(name: {_name}, signed: {signed}, decimal: {decimalNum})"; } } static partial class TextInputUtils { internal static TextAffinity? _toTextAffinity(string affinity) { switch (affinity) { case "TextAffinity.downstream": return TextAffinity.downstream; case "TextAffinity.upstream": return TextAffinity.upstream; } return null; } internal static TextInputAction _toTextInputAction(string action) { switch (action) { case "TextInputAction.none": return TextInputAction.none; case "TextInputAction.unspecified": return TextInputAction.unspecified; case "TextInputAction.go": return TextInputAction.go; case "TextInputAction.search": return TextInputAction.search; case "TextInputAction.send": return TextInputAction.send; case "TextInputAction.next": return TextInputAction.next; case "TextInputAction.previuos": return TextInputAction.previous; case "TextInputAction.continue_action": return TextInputAction.continueAction; case "TextInputAction.join": return TextInputAction.join; case "TextInputAction.route": return TextInputAction.route; case "TextInputAction.emergencyCall": return TextInputAction.emergencyCall; case "TextInputAction.done": return TextInputAction.done; case "TextInputAction.newline": return TextInputAction.newline; } throw new UIWidgetsError("Unknown text input action: $action"); } public static FloatingCursorDragState _toTextCursorAction(string state) { switch (state) { case "FloatingCursorDragState.start": return FloatingCursorDragState.Start; case "FloatingCursorDragState.update": return FloatingCursorDragState.Update; case "FloatingCursorDragState.end": return FloatingCursorDragState.End; } throw new UIWidgetsError("Unknown text cursor action: $state"); } public static RawFloatingCursorPoint _toTextPoint(FloatingCursorDragState state, Dictionary encoded) { D.assert(encoded.getOrDefault("X") != null, () => "You must provide a value for the horizontal location of the floating cursor."); D.assert(encoded.getOrDefault("Y") != null, () => "You must provide a value for the vertical location of the floating cursor."); Offset offset = state == FloatingCursorDragState.Update ? new Offset(encoded["X"] ?? 0.0f, encoded["Y"] ?? 0.0f) : new Offset(0, 0); return new RawFloatingCursorPoint(offset: offset, state: state); } } public class TextEditingValue : IEquatable { public readonly string text; public readonly TextSelection selection; public readonly TextRange composing; static JSONNode defaultIndexNode = new JSONNumber(-1); public TextEditingValue(string text = "", TextSelection selection = null, TextRange composing = null) { this.text = text; this.composing = composing ?? TextRange.empty; if (selection != null && selection.start >= 0 && selection.end >= 0) { // handle surrogate pair emoji, which takes 2 utf16 chars // if selection cuts in the middle of the emoji, move it to the end int start = selection.start, end = selection.end; if (start < text.Length && char.IsLowSurrogate(text[start])) { start++; } if (end < text.Length && char.IsLowSurrogate(text[end])) { end++; } this.selection = selection.copyWith(start, end); } else { this.selection = TextSelection.collapsed(-1); } } public static TextEditingValue fromJson(JSONObject json) { TextAffinity? affinity = TextInputUtils._toTextAffinity(json["selectionAffinity"].Value); return new TextEditingValue( text: json["text"].Value, selection: new TextSelection( baseOffset: json.GetValueOrDefault("selectionBase", defaultIndexNode).AsInt, extentOffset: json.GetValueOrDefault("selectionExtent", defaultIndexNode).AsInt, affinity: affinity != null ? affinity.Value : TextAffinity.downstream, isDirectional: json["selectionIsDirectional"].AsBool ), composing: new TextRange( start: json.GetValueOrDefault("composingBase", defaultIndexNode).AsInt, end: json.GetValueOrDefault("composingExtent", defaultIndexNode).AsInt ) ); } public TextEditingValue copyWith(string text = null, TextSelection selection = null, TextRange composing = null) { return new TextEditingValue( text ?? this.text, selection ?? this.selection, composing ?? this.composing ); } public TextEditingValue insert(string text) { string newText; TextSelection newSelection; if (string.IsNullOrEmpty(text)) { return this; } var selection = this.selection; if (selection.start < 0) { selection = TextSelection.collapsed(0, this.selection.affinity); } newText = selection.textBefore(this.text) + text + selection.textAfter(this.text); newSelection = TextSelection.collapsed(selection.start + text.Length); return new TextEditingValue( text: newText, selection: newSelection, composing: TextRange.empty ); } public TextEditingValue deleteSelection(bool backDelete = true) { if (selection.isCollapsed) { if (backDelete) { if (selection.start == 0) { return this; } if (char.IsHighSurrogate(text[selection.start - 1])) { return copyWith( text: text.Substring(0, selection.start - 1) + text.Substring(selection.start + 1), selection: TextSelection.collapsed(selection.start - 1), composing: TextRange.empty); } if (char.IsLowSurrogate(text[selection.start - 1])) { D.assert(selection.start > 1); return copyWith( text: text.Substring(0, selection.start - 2) + selection.textAfter(text), selection: TextSelection.collapsed(selection.start - 2), composing: TextRange.empty); } return copyWith( text: text.Substring(0, selection.start - 1) + selection.textAfter(text), selection: TextSelection.collapsed(selection.start - 1), composing: TextRange.empty); } if (selection.start >= text.Length) { return this; } return copyWith(text: text.Substring(0, selection.start) + text.Substring(selection.start + 1), composing: TextRange.empty); } else { var newText = selection.textBefore(text) + selection.textAfter(text); return copyWith(text: newText, selection: TextSelection.collapsed(selection.start), composing: TextRange.empty); } } public TextEditingValue moveLeft() { return moveSelection(-1); } public TextEditingValue moveRight() { return moveSelection(1); } public TextEditingValue extendLeft() { return moveExtent(-1); } public TextEditingValue extendRight() { return moveExtent(1); } public TextEditingValue moveExtent(int move) { int offset = selection.extentOffset + move; offset = Mathf.Max(0, offset); offset = Mathf.Min(offset, text.Length); return copyWith(selection: selection.copyWith(extentOffset: offset)); } public TextEditingValue moveSelection(int move) { int offset = selection.baseOffset + move; offset = Mathf.Max(0, offset); offset = Mathf.Min(offset, text.Length); return copyWith(selection: TextSelection.collapsed(offset, affinity: selection.affinity)); } public TextEditingValue compose(string composeText) { D.assert(!string.IsNullOrEmpty(composeText)); var composeStart = composing == TextRange.empty ? selection.start : composing.start; var lastComposeEnd = composing == TextRange.empty ? selection.end : composing.end; composeStart = Mathf.Clamp(composeStart, 0, text.Length); lastComposeEnd = Mathf.Clamp(lastComposeEnd, 0, text.Length); var newText = text.Substring(0, composeStart) + composeText + text.Substring(lastComposeEnd); var componseEnd = composeStart + composeText.Length; return new TextEditingValue( text: newText, selection: TextSelection.collapsed(componseEnd), composing: new TextRange(composeStart, componseEnd) ); } public TextEditingValue clearCompose() { if (composing == TextRange.empty) { return this; } return new TextEditingValue( text: text.Substring(0, composing.start) + text.Substring(composing.end), selection: TextSelection.collapsed(composing.start), composing: TextRange.empty ); } public static readonly TextEditingValue empty = new TextEditingValue(); public bool Equals(TextEditingValue other) { if (ReferenceEquals(null, other)) { return false; } if (ReferenceEquals(this, other)) { return true; } return string.Equals(text, other.text) && Equals(selection, other.selection) && Equals(composing, other.composing); } public override bool Equals(object obj) { if (ReferenceEquals(null, obj)) { return false; } if (ReferenceEquals(this, obj)) { return true; } if (obj.GetType() != GetType()) { return false; } return Equals((TextEditingValue) obj); } public override int GetHashCode() { unchecked { var hashCode = (text != null ? text.GetHashCode() : 0); hashCode = (hashCode * 397) ^ (selection != null ? selection.GetHashCode() : 0); hashCode = (hashCode * 397) ^ (composing != null ? composing.GetHashCode() : 0); return hashCode; } } public static bool operator ==(TextEditingValue left, TextEditingValue right) { return Equals(left, right); } public static bool operator !=(TextEditingValue left, TextEditingValue right) { return !Equals(left, right); } public override string ToString() { return $"Text: {text}, Selection: {selection}, Composing: {composing}"; } public JSONNode toJson() { var json = new JSONObject(); json["text"] = text; json["selectionBase"] = selection.baseOffset; json["selectionExtent"] = selection.extentOffset; json["selectionAffinity"] = selection.affinity.ToString(); json["selectionIsDirectional"] = selection.isDirectional; json["composingBase"] = composing.start; json["composingExtent"] = composing.end; return json; } } public interface TextSelectionDelegate { TextEditingValue textEditingValue { get; set; } void hideToolbar(); void bringIntoView(TextPosition textPosition); } public interface TextInputClient { void updateEditingValue(TextEditingValue value, bool isIMEInput); void performAction(TextInputAction action); void updateFloatingCursor(RawFloatingCursorPoint point); RawInputKeyResponse globalInputKeyHandler(RawKeyEvent evt); } public enum TextInputAction { none, unspecified, done, go, search, send, next, previous, continueAction, join, route, emergencyCall, newline } public enum TextCapitalization { words, sentences, characters, none } class TextInputConfiguration { public TextInputConfiguration(TextInputType inputType = null, bool obscureText = false, bool autocorrect = true, TextInputAction inputAction = TextInputAction.done, Brightness keyboardAppearance = Brightness.light, TextCapitalization textCapitalization = TextCapitalization.none, bool unityTouchKeyboard = false) { this.inputType = inputType ?? TextInputType.text; this.inputAction = inputAction; this.obscureText = obscureText; this.autocorrect = autocorrect; this.textCapitalization = textCapitalization; this.keyboardAppearance = keyboardAppearance; this.unityTouchKeyboard = unityTouchKeyboard; } public readonly TextInputType inputType; public readonly bool obscureText; public readonly bool autocorrect; public readonly TextInputAction inputAction; public readonly TextCapitalization textCapitalization; public readonly Brightness keyboardAppearance; public readonly bool unityTouchKeyboard; public JSONNode toJson() { var json = new JSONObject(); json["inputType"] = inputType.toJson(); json["obscureText"] = obscureText; json["autocorrect"] = autocorrect; json["inputAction"] = $"TextInputAction.{inputAction.ToString()}"; json["unityTouchKeyboard"] = unityTouchKeyboard; json["textCapitalization"] = $"TextCapitalization.{textCapitalization.ToString()}"; json["keyboardAppearance"] = $"Brightness.{keyboardAppearance.ToString()}"; return json; } } public class TextInputConnection { internal TextInputConnection(TextInputClient client) { D.assert(client != null); _window = Window.instance; _client = client; _id = _nextId++; } public bool attached { get { return TextInput._currentConnection == this; } } public void setEditingState(TextEditingValue value) { D.assert(attached); TextInput.keyboardDelegate.setEditingState(value); } public void setIMEPos(Offset imeGlobalPos) { D.assert(attached); D.assert(imeGlobalPos != null); D.assert(imeRequired()); TextInput.keyboardDelegate.setIMEPos(imeGlobalPos); } public bool imeRequired() { return TextInput.keyboardDelegate != null && TextInput.keyboardDelegate.imeRequired(); } public void close() { if (attached) { TextInput.keyboardDelegate.clearClient(); TextInput._currentConnection = null; Input.imeCompositionMode = IMECompositionMode.Auto; TextInput._scheduleHide(); } D.assert(!attached); } public void show() { D.assert(attached); Input.imeCompositionMode = IMECompositionMode.On; TextInput.keyboardDelegate.show(); } static int _nextId = 1; internal readonly int _id; internal readonly TextInputClient _client; internal readonly Window _window; TouchScreenKeyboard _keyboard; } class TextInput { static internal TextInputConnection _currentConnection; static internal KeyboardDelegate keyboardDelegate; public TextInput() { } public static TextInputConnection attach(TextInputClient client, TextInputConfiguration configuration) { D.assert(client != null); var connection = new TextInputConnection(client); _currentConnection = connection; if (keyboardDelegate != null) { keyboardDelegate.Dispose(); } if (Application.isEditor) { keyboardDelegate = new DefaultKeyboardDelegate(); } else { #if UNITY_IOS || UNITY_ANDROID if (configuration.unityTouchKeyboard) { keyboardDelegate = new UnityTouchScreenKeyboardDelegate(); } else { keyboardDelegate = new UIWidgetsTouchScreenKeyboardDelegate(); } #elif UNITY_WEBGL keyboardDelegate = new UIWidgetsWebGLKeyboardDelegate(); #else keyboardDelegate = new DefaultKeyboardDelegate(); #endif } keyboardDelegate.setClient(connection._id, configuration); return connection; } internal static void Update() { if (_currentConnection != null && _currentConnection._window == Window.instance) { (keyboardDelegate as TextInputUpdateListener)?.Update(); } } internal static void OnGUI() { if (_currentConnection != null && _currentConnection._window == Window.instance) { (keyboardDelegate as TextInputOnGUIListener)?.OnGUI(); } } internal static void _updateEditingState(int client, TextEditingValue value, bool isIMEInput = false) { if (_currentConnection == null) { return; } if (client != _currentConnection._id) { return; } _currentConnection._client.updateEditingValue(value, isIMEInput); } internal static void _performAction(int client, TextInputAction action) { if (_currentConnection == null) { return; } if (client != _currentConnection._id) { return; } _currentConnection._client.performAction(action); } internal static RawInputKeyResponse _handleGlobalInputKey(int client, RawKeyEvent evt) { if (_currentConnection == null) { return RawInputKeyResponse.convert(evt); } if (client != _currentConnection._id) { return RawInputKeyResponse.convert(evt); } return _currentConnection._client.globalInputKeyHandler(evt); } static bool _hidePending = false; static internal void _scheduleHide() { if (_hidePending) { return; } _hidePending = true; Window.instance.scheduleMicrotask(() => { _hidePending = false; if (_currentConnection == null) { keyboardDelegate.hide(); } }); } } }