using System; using System.Collections.Generic; using RSG.Promises; using Unity.UIWidgets.animation; using Unity.UIWidgets.foundation; using Unity.UIWidgets.gestures; using Unity.UIWidgets.painting; using Unity.UIWidgets.rendering; using Unity.UIWidgets.service; using Unity.UIWidgets.ui; using Unity.UIWidgets.widgets; using UnityEngine; using Canvas = Unity.UIWidgets.ui.Canvas; using Color = Unity.UIWidgets.ui.Color; using Rect = Unity.UIWidgets.ui.Rect; using TextStyle = Unity.UIWidgets.painting.TextStyle; using Transform = Unity.UIWidgets.widgets.Transform; namespace Unity.UIWidgets.material { class InputDecoratorConstants { public static readonly TimeSpan _kTransitionDuration = new TimeSpan(0, 0, 0, 0, 200); public static readonly Curve _kTransitionCurve = Curves.fastOutSlowIn; } class _InputBorderGap : ChangeNotifier, IEquatable<_InputBorderGap> { float _start; public float start { get { return this._start; } set { if (value != this._start) { this._start = value; this.notifyListeners(); } } } float _extent = 0.0f; public float extent { get { return this._extent; } set { if (value != this._extent) { this._extent = value; this.notifyListeners(); } } } public bool Equals(_InputBorderGap other) { if (ReferenceEquals(null, other)) { return false; } if (ReferenceEquals(this, other)) { return true; } return this.start == other.start && this.extent == other._extent; } public override bool Equals(object obj) { if (ReferenceEquals(null, obj)) { return false; } if (ReferenceEquals(this, obj)) { return true; } if (obj.GetType() != this.GetType()) { return false; } return this.Equals((_InputBorderGap) obj); } public override int GetHashCode() { unchecked { return (this._start.GetHashCode() * 397) ^ this._extent.GetHashCode(); } } public static bool operator ==(_InputBorderGap left, _InputBorderGap right) { return Equals(left, right); } public static bool operator !=(_InputBorderGap left, _InputBorderGap right) { return !Equals(left, right); } } class _InputBorderTween : Tween { public _InputBorderTween(InputBorder begin = null, InputBorder end = null) : base(begin: begin, end: end) { } public override InputBorder lerp(float t) { return (InputBorder) ShapeBorder.lerp(this.begin, this.end, t); } } class _InputBorderPainter : AbstractCustomPainter { public _InputBorderPainter( Listenable repaint, Animation borderAnimation = null, _InputBorderTween border = null, Animation gapAnimation = null, _InputBorderGap gap = null, Color fillColor = null ) : base(repaint: repaint) { this.borderAnimation = borderAnimation; this.border = border; this.gapAnimation = gapAnimation; this.gap = gap; this.fillColor = fillColor; } public readonly Animation borderAnimation; public readonly _InputBorderTween border; public readonly Animation gapAnimation; public readonly _InputBorderGap gap; public readonly Color fillColor; public override void paint(Canvas canvas, Size size) { InputBorder borderValue = this.border.evaluate(this.borderAnimation); Rect canvasRect = Offset.zero & size; if (this.fillColor.alpha > 0) { Paint paint = new Paint(); paint.color = this.fillColor; paint.style = PaintingStyle.fill; canvas.drawPath( borderValue.getOuterPath(canvasRect), paint ); } borderValue.paint( canvas, canvasRect, gapStart: this.gap.start, gapExtent: this.gap.extent, gapPercentage: this.gapAnimation.value ); } public override bool shouldRepaint(CustomPainter _oldPainter) { _InputBorderPainter oldPainter = _oldPainter as _InputBorderPainter; return this.borderAnimation != oldPainter.borderAnimation || this.gapAnimation != oldPainter.gapAnimation || this.border != oldPainter.border || this.gap != oldPainter.gap; } } class _BorderContainer : StatefulWidget { public _BorderContainer( Key key = null, InputBorder border = null, _InputBorderGap gap = null, Animation gapAnimation = null, Color fillColor = null, Widget child = null ) : base(key: key) { D.assert(border != null); D.assert(gap != null); D.assert(fillColor != null); this.border = border; this.gap = gap; this.gapAnimation = gapAnimation; this.fillColor = fillColor; this.child = child; } public readonly InputBorder border; public readonly _InputBorderGap gap; public readonly Animation gapAnimation; public readonly Color fillColor; public readonly Widget child; public override State createState() { return new _BorderContainerState(); } } class _BorderContainerState : SingleTickerProviderStateMixin<_BorderContainer> { AnimationController _controller; Animation _borderAnimation; _InputBorderTween _border; public override void initState() { base.initState(); this._controller = new AnimationController( duration: InputDecoratorConstants._kTransitionDuration, vsync: this ); this._borderAnimation = new CurvedAnimation( parent: this._controller, curve: InputDecoratorConstants._kTransitionCurve ); this._border = new _InputBorderTween( begin: this.widget.border, end: this.widget.border ); } public override void dispose() { this._controller.dispose(); base.dispose(); } public override void didUpdateWidget(StatefulWidget _oldWidget) { _BorderContainer oldWidget = _oldWidget as _BorderContainer; base.didUpdateWidget(oldWidget); if (this.widget.border != oldWidget.border) { this._border = new _InputBorderTween( begin: oldWidget.border, end: this.widget.border ); this._controller.setValue(0.0f); this._controller.forward(); } } public override Widget build(BuildContext context) { return new CustomPaint( foregroundPainter: new _InputBorderPainter( repaint: ListenableUtils.merge(new List {this._borderAnimation, this.widget.gap}), borderAnimation: this._borderAnimation, border: this._border, gapAnimation: this.widget.gapAnimation, gap: this.widget.gap, fillColor: this.widget.fillColor ), child: this.widget.child ); } } class _Shaker : AnimatedWidget { public _Shaker( Key key = null, Animation animation = null, Widget child = null ) : base(key: key, listenable: animation) { this.child = child; } public readonly Widget child; public Animation animation { get { return (Animation) this.listenable; } } public float translateX { get { const float shakeDelta = 4.0f; float t = this.animation.value; if (t <= 0.25f) { return -t * shakeDelta; } else if (t < 0.75f) { return (t - 0.5f) * shakeDelta; } else { return (1.0f - t) * 4.0f * shakeDelta; } } } protected internal override Widget build(BuildContext context) { return new Transform( transform: new Matrix4().translationValues(this.translateX, 0, 0), child: this.child ); } } class _HelperError : StatefulWidget { public _HelperError( Key key = null, TextAlign? textAlign = null, string helperText = null, TextStyle helperStyle = null, string errorText = null, TextStyle errorStyle = null, int? errorMaxLines = null ) : base(key: key) { this.textAlign = textAlign; this.helperText = helperText; this.helperStyle = helperStyle; this.errorText = errorText; this.errorStyle = errorStyle; this.errorMaxLines = errorMaxLines; } public readonly TextAlign? textAlign; public readonly string helperText; public readonly TextStyle helperStyle; public readonly string errorText; public readonly TextStyle errorStyle; public readonly int? errorMaxLines; public override State createState() { return new _HelperErrorState(); } } class _HelperErrorState : SingleTickerProviderStateMixin<_HelperError> { static readonly Widget empty = new SizedBox(); AnimationController _controller; Widget _helper; Widget _error; public override void initState() { base.initState(); this._controller = new AnimationController( duration: InputDecoratorConstants._kTransitionDuration, vsync: this ); if (this.widget.errorText != null) { this._error = this._buildError(); this._controller.setValue(1.0f); } else if (this.widget.helperText != null) { this._helper = this._buildHelper(); } this._controller.addListener(this._handleChange); } public override void dispose() { this._controller.dispose(); base.dispose(); } void _handleChange() { this.setState(() => { }); } public override void didUpdateWidget(StatefulWidget _old) { base.didUpdateWidget(_old); _HelperError old = _old as _HelperError; string newErrorText = this.widget.errorText; string newHelperText = this.widget.helperText; string oldErrorText = old.errorText; string oldHelperText = old.helperText; bool errorTextStateChanged = (newErrorText != null) != (oldErrorText != null); bool helperTextStateChanged = newErrorText == null && (newHelperText != null) != (oldHelperText != null); if (errorTextStateChanged || helperTextStateChanged) { if (newErrorText != null) { this._error = this._buildError(); this._controller.forward(); } else if (newHelperText != null) { this._helper = this._buildHelper(); this._controller.reverse(); } else { this._controller.reverse(); } } } Widget _buildHelper() { D.assert(this.widget.helperText != null); return new Opacity( opacity: 1.0f - this._controller.value, child: new Text( this.widget.helperText, style: this.widget.helperStyle, textAlign: this.widget.textAlign, overflow: TextOverflow.ellipsis ) ); } Widget _buildError() { D.assert(this.widget.errorText != null); return new Opacity( opacity: this._controller.value, child: new FractionalTranslation( translation: new OffsetTween( begin: new Offset(0.0f, -0.25f), end: new Offset(0.0f, 0.0f) ).evaluate(this._controller.view), child: new Text( this.widget.errorText, style: this.widget.errorStyle, textAlign: this.widget.textAlign, overflow: TextOverflow.ellipsis, maxLines: this.widget.errorMaxLines ) ) ); } public override Widget build(BuildContext context) { if (this._controller.isDismissed) { this._error = null; if (this.widget.helperText != null) { return this._helper = this._buildHelper(); } else { this._helper = null; return empty; } } if (this._controller.isCompleted) { this._helper = null; if (this.widget.errorText != null) { return this._error = this._buildError(); } else { this._error = null; return empty; } } if (this._helper == null && this.widget.errorText != null) { return this._buildError(); } if (this._error == null && this.widget.helperText != null) { return this._buildHelper(); } if (this.widget.errorText != null) { return new Stack( children: new List { new Opacity( opacity: 1.0f - this._controller.value, child: this._helper ), this._buildError(), } ); } if (this.widget.helperText != null) { return new Stack( children: new List { this._buildHelper(), new Opacity( opacity: this._controller.value, child: this._error ) } ); } return empty; } } enum _DecorationSlot { icon, input, label, hint, prefix, suffix, prefixIcon, suffixIcon, helperError, counter, container } class _Decoration : IEquatable<_Decoration> { public _Decoration( EdgeInsets contentPadding, bool isCollapsed, float floatingLabelHeight, float floatingLabelProgress, InputBorder border = null, _InputBorderGap borderGap = null, Widget icon = null, Widget input = null, Widget label = null, Widget hint = null, Widget prefix = null, Widget suffix = null, Widget prefixIcon = null, Widget suffixIcon = null, Widget helperError = null, Widget counter = null, Widget container = null, bool? alignLabelWithHint = null ) { D.assert(contentPadding != null); this.contentPadding = contentPadding; this.isCollapsed = isCollapsed; this.floatingLabelHeight = floatingLabelHeight; this.floatingLabelProgress = floatingLabelProgress; this.border = border; this.borderGap = borderGap; this.icon = icon; this.input = input; this.label = label; this.hint = hint; this.prefix = prefix; this.suffix = suffix; this.prefixIcon = prefixIcon; this.suffixIcon = suffixIcon; this.helperError = helperError; this.counter = counter; this.container = container; this.alignLabelWithHint = alignLabelWithHint; } public readonly EdgeInsets contentPadding; public readonly bool isCollapsed; public readonly float floatingLabelHeight; public readonly float floatingLabelProgress; public readonly InputBorder border; public readonly _InputBorderGap borderGap; public readonly bool? alignLabelWithHint; public readonly Widget icon; public readonly Widget input; public readonly Widget label; public readonly Widget hint; public readonly Widget prefix; public readonly Widget suffix; public readonly Widget prefixIcon; public readonly Widget suffixIcon; public readonly Widget helperError; public readonly Widget counter; public readonly Widget container; public bool Equals(_Decoration other) { if (ReferenceEquals(null, other)) { return false; } if (ReferenceEquals(this, other)) { return true; } return Equals(this.contentPadding, other.contentPadding) && this.isCollapsed == other.isCollapsed && this.floatingLabelHeight.Equals(other.floatingLabelHeight) && this.floatingLabelProgress.Equals(other.floatingLabelProgress) && Equals(this.border, other.border) && Equals(this.borderGap, other.borderGap) && Equals(this.icon, other.icon) && Equals(this.input, other.input) && Equals(this.label, other.label) && Equals(this.hint, other.hint) && Equals(this.prefix, other.prefix) && Equals(this.suffix, other.suffix) && Equals(this.prefixIcon, other.prefixIcon) && Equals(this.suffixIcon, other.suffixIcon) && Equals(this.helperError, other.helperError) && Equals(this.counter, other.counter) && Equals(this.container, other.container) && Equals(this.alignLabelWithHint, other.alignLabelWithHint); } public override bool Equals(object obj) { if (ReferenceEquals(null, obj)) { return false; } if (ReferenceEquals(this, obj)) { return true; } if (obj.GetType() != this.GetType()) { return false; } return this.Equals((_Decoration) obj); } public override int GetHashCode() { unchecked { var hashCode = (this.contentPadding != null ? this.contentPadding.GetHashCode() : 0); hashCode = (hashCode * 397) ^ this.isCollapsed.GetHashCode(); hashCode = (hashCode * 397) ^ this.floatingLabelHeight.GetHashCode(); hashCode = (hashCode * 397) ^ this.floatingLabelProgress.GetHashCode(); hashCode = (hashCode * 397) ^ (this.border != null ? this.border.GetHashCode() : 0); hashCode = (hashCode * 397) ^ (this.borderGap != null ? this.borderGap.GetHashCode() : 0); hashCode = (hashCode * 397) ^ (this.icon != null ? this.icon.GetHashCode() : 0); hashCode = (hashCode * 397) ^ (this.input != null ? this.input.GetHashCode() : 0); hashCode = (hashCode * 397) ^ (this.label != null ? this.label.GetHashCode() : 0); hashCode = (hashCode * 397) ^ (this.hint != null ? this.hint.GetHashCode() : 0); hashCode = (hashCode * 397) ^ (this.prefix != null ? this.prefix.GetHashCode() : 0); hashCode = (hashCode * 397) ^ (this.suffix != null ? this.suffix.GetHashCode() : 0); hashCode = (hashCode * 397) ^ (this.prefixIcon != null ? this.prefixIcon.GetHashCode() : 0); hashCode = (hashCode * 397) ^ (this.suffixIcon != null ? this.suffixIcon.GetHashCode() : 0); hashCode = (hashCode * 397) ^ (this.helperError != null ? this.helperError.GetHashCode() : 0); hashCode = (hashCode * 397) ^ (this.counter != null ? this.counter.GetHashCode() : 0); hashCode = (hashCode * 397) ^ (this.container != null ? this.container.GetHashCode() : 0); hashCode = (hashCode * 397) ^ (this.alignLabelWithHint != null ? this.alignLabelWithHint.GetHashCode() : 0); return hashCode; } } public static bool operator ==(_Decoration left, _Decoration right) { return Equals(left, right); } public static bool operator !=(_Decoration left, _Decoration right) { return !Equals(left, right); } } class _RenderDecorationLayout { public _RenderDecorationLayout( Dictionary boxToBaseline = null, float? inputBaseline = null, float? outlineBaseline = null, float? subtextBaseline = null, float? containerHeight = null, float? subtextHeight = null ) { this.boxToBaseline = boxToBaseline; this.inputBaseline = inputBaseline; this.outlineBaseline = outlineBaseline; this.subtextBaseline = subtextBaseline; this.containerHeight = containerHeight; this.subtextHeight = subtextHeight; } public readonly Dictionary boxToBaseline; public readonly float? inputBaseline; public readonly float? outlineBaseline; public readonly float? subtextBaseline; public readonly float? containerHeight; public readonly float? subtextHeight; } class _RenderDecoration : RenderBox { public _RenderDecoration( _Decoration decoration, TextBaseline? textBaseline, bool isFocused, bool expands ) { D.assert(decoration != null); D.assert(textBaseline != null); this._decoration = decoration; this._textBaseline = textBaseline; this._isFocused = isFocused; this._expands = expands; } public const float subtextGap = 8.0f; public readonly Dictionary<_DecorationSlot, RenderBox> slotToChild = new Dictionary<_DecorationSlot, RenderBox>(); public readonly Dictionary childToSlot = new Dictionary(); RenderBox _updateChild(RenderBox oldChild, RenderBox newChild, _DecorationSlot slot) { if (oldChild != null) { this.dropChild(oldChild); this.childToSlot.Remove(oldChild); this.slotToChild.Remove(slot); } if (newChild != null) { this.childToSlot[newChild] = slot; this.slotToChild[slot] = newChild; this.adoptChild(newChild); } return newChild; } RenderBox _icon; public RenderBox icon { get { return this._icon; } set { this._icon = this._updateChild(this._icon, value, _DecorationSlot.icon); } } RenderBox _input; public RenderBox input { get { return this._input; } set { this._input = this._updateChild(this._input, value, _DecorationSlot.input); } } RenderBox _label; public RenderBox label { get { return this._label; } set { this._label = this._updateChild(this._label, value, _DecorationSlot.label); } } RenderBox _hint; public RenderBox hint { get { return this._hint; } set { this._hint = this._updateChild(this._hint, value, _DecorationSlot.hint); } } RenderBox _prefix; public RenderBox prefix { get { return this._prefix; } set { this._prefix = this._updateChild(this._prefix, value, _DecorationSlot.prefix); } } RenderBox _suffix; public RenderBox suffix { get { return this._suffix; } set { this._suffix = this._updateChild(this._suffix, value, _DecorationSlot.suffix); } } RenderBox _prefixIcon; public RenderBox prefixIcon { get { return this._prefixIcon; } set { this._prefixIcon = this._updateChild(this._prefixIcon, value, _DecorationSlot.prefixIcon); } } RenderBox _suffixIcon; public RenderBox suffixIcon { get { return this._suffixIcon; } set { this._suffixIcon = this._updateChild(this._suffixIcon, value, _DecorationSlot.suffixIcon); } } RenderBox _helperError; public RenderBox helperError { get { return this._helperError; } set { this._helperError = this._updateChild(this._helperError, value, _DecorationSlot.helperError); } } RenderBox _counter; public RenderBox counter { get { return this._counter; } set { this._counter = this._updateChild(this._counter, value, _DecorationSlot.counter); } } RenderBox _container; public RenderBox container { get { return this._container; } set { this._container = this._updateChild(this._container, value, _DecorationSlot.container); } } IEnumerable _children { get { if (this.icon != null) { yield return this.icon; } if (this.input != null) { yield return this.input; } if (this.prefixIcon != null) { yield return this.prefixIcon; } if (this.suffixIcon != null) { yield return this.suffixIcon; } if (this.prefix != null) { yield return this.prefix; } if (this.suffix != null) { yield return this.suffix; } if (this.label != null) { yield return this.label; } if (this.hint != null) { yield return this.hint; } if (this.helperError != null) { yield return this.helperError; } if (this.counter != null) { yield return this.counter; } if (this.container != null) { yield return this.container; } } } public _Decoration decoration { get { return this._decoration; } set { D.assert(value != null); if (this._decoration == value) { return; } this._decoration = value; this.markNeedsLayout(); } } _Decoration _decoration; public TextBaseline? textBaseline { get { return this._textBaseline; } set { D.assert(value != null); if (this._textBaseline == value) { return; } this._textBaseline = value; this.markNeedsLayout(); } } TextBaseline? _textBaseline; public bool isFocused { get { return this._isFocused; } set { if (this._isFocused == value) { return; } this._isFocused = value; } } bool _isFocused; public bool expands { get { return this._expands; } set { if (this._expands == value) { return; } this._expands = value; this.markNeedsLayout(); } } bool _expands = false; public override void attach(object owner) { base.attach(owner); foreach (RenderBox child in this._children) { child.attach(owner); } } public override void detach() { base.detach(); foreach (RenderBox child in this._children) { child.detach(); } } public override void redepthChildren() { this._children.Each(this.redepthChild); } public override void visitChildren(RenderObjectVisitor visitor) { this._children.Each((child) => { visitor(child); }); } public override List debugDescribeChildren() { List value = new List { }; void add(RenderBox child, string name) { if (child != null) { value.Add(child.toDiagnosticsNode(name: name)); } } add(this.icon, "icon"); add(this.input, "input"); add(this.label, "label"); add(this.hint, "hint"); add(this.prefix, "prefix"); add(this.suffix, "suffix"); add(this.prefixIcon, "prefixIcon"); add(this.suffixIcon, "suffixIcon"); add(this.helperError, "helperError"); add(this.counter, "counter"); add(this.container, "container"); return value; } protected override bool sizedByParent { get { return false; } } static float _minWidth(RenderBox box, float height) { return box == null ? 0.0f : box.getMinIntrinsicWidth(height); } static float _maxWidth(RenderBox box, float height) { return box == null ? 0.0f : box.getMaxIntrinsicWidth(height); } static float _minHeight(RenderBox box, float width) { return box == null ? 0.0f : box.getMinIntrinsicHeight(width); } static Size _boxSize(RenderBox box) { return box == null ? Size.zero : box.size; } static BoxParentData _boxParentData(RenderBox box) { return (BoxParentData) box.parentData; } public EdgeInsets contentPadding { get { return this.decoration.contentPadding; } } float _layoutLineBox(RenderBox box, BoxConstraints constraints) { if (box == null) { return 0.0f; } box.layout(constraints, parentUsesSize: true); float baseline = box.getDistanceToBaseline(this.textBaseline.Value).Value; D.assert(baseline >= 0.0f); return baseline; } _RenderDecorationLayout _layout(BoxConstraints layoutConstraints) { Dictionary boxToBaseline = new Dictionary(); BoxConstraints boxConstraints = layoutConstraints.loosen(); if (this.prefix != null) { boxToBaseline[this.prefix] = this._layoutLineBox(this.prefix, boxConstraints); } if (this.suffix != null) { boxToBaseline[this.suffix] = this._layoutLineBox(this.suffix, boxConstraints); } if (this.icon != null) { boxToBaseline[this.icon] = this._layoutLineBox(this.icon, boxConstraints); } if (this.prefixIcon != null) { boxToBaseline[this.prefixIcon] = this._layoutLineBox(this.prefixIcon, boxConstraints); } if (this.suffixIcon != null) { boxToBaseline[this.suffixIcon] = this._layoutLineBox(this.suffixIcon, boxConstraints); } float inputWidth = Math.Max(0.0f, this.constraints.maxWidth - ( _boxSize(this.icon).width + this.contentPadding.left + _boxSize(this.prefixIcon).width + _boxSize(this.prefix).width + _boxSize(this.suffix).width + _boxSize(this.suffixIcon).width + this.contentPadding.right)); if (this.label != null) { boxToBaseline[this.label] = this._layoutLineBox(this.label, boxConstraints.copyWith(maxWidth: inputWidth) ); } if (this.hint != null) { boxToBaseline[this.hint] = this._layoutLineBox(this.hint, boxConstraints.copyWith(minWidth: inputWidth, maxWidth: inputWidth) ); } if (this.counter != null) { boxToBaseline[this.counter] = this._layoutLineBox(this.counter, boxConstraints); } if (this.helperError != null) { boxToBaseline[this.helperError] = this._layoutLineBox(this.helperError, boxConstraints.copyWith( maxWidth: Math.Max(0.0f, boxConstraints.maxWidth - _boxSize(this.icon).width - _boxSize(this.counter).width - this.contentPadding.horizontal ) ) ); } float labelHeight = this.label == null ? 0 : this.decoration.floatingLabelHeight; float topHeight = this.decoration.border.isOutline ? Math.Max(labelHeight - boxToBaseline.getOrDefault(this.label, 0), 0) : labelHeight; float counterHeight = this.counter == null ? 0 : boxToBaseline.getOrDefault(this.counter, 0) + subtextGap; bool helperErrorExists = this.helperError?.size != null && this.helperError.size.height > 0; float helperErrorHeight = !helperErrorExists ? 0 : this.helperError.size.height + subtextGap; float bottomHeight = Math.Max( counterHeight, helperErrorHeight ); if (this.input != null) { boxToBaseline[this.input] = this._layoutLineBox(this.input, boxConstraints.deflate(EdgeInsets.only( top: this.contentPadding.top + topHeight, bottom: this.contentPadding.bottom + bottomHeight )).copyWith( minWidth: inputWidth, maxWidth: inputWidth ) ); } // The field can be occupied by a hint or by the input itself float hintHeight = this.hint == null ? 0 : this.hint.size.height; float inputDirectHeight = this.input == null ? 0 : this.input.size.height; float inputHeight = Math.Max(hintHeight, inputDirectHeight); float inputInternalBaseline = Math.Max( boxToBaseline.getOrDefault(this.input, 0.0f), boxToBaseline.getOrDefault(this.hint, 0.0f) ); // Calculate the amount that prefix/suffix affects height above and below // the input. float prefixHeight = this.prefix == null ? 0 : this.prefix.size.height; float suffixHeight = this.suffix == null ? 0 : this.suffix.size.height; float fixHeight = Math.Max( boxToBaseline.getOrDefault(this.prefix, 0.0f), boxToBaseline.getOrDefault(this.suffix, 0.0f) ); float fixAboveInput = Math.Max(0, fixHeight - inputInternalBaseline); float fixBelowBaseline = Math.Max( prefixHeight - boxToBaseline.getOrDefault(this.prefix, 0.0f), suffixHeight - boxToBaseline.getOrDefault(this.suffix, 0.0f) ); float fixBelowInput = Math.Max( 0, fixBelowBaseline - (inputHeight - inputInternalBaseline) ); // Calculate the height of the input text container. float prefixIconHeight = this.prefixIcon == null ? 0 : this.prefixIcon.size.height; float suffixIconHeight = this.suffixIcon == null ? 0 : this.suffixIcon.size.height; float fixIconHeight = Math.Max(prefixIconHeight, suffixIconHeight); float contentHeight = Math.Max( fixIconHeight, topHeight + this.contentPadding.top + fixAboveInput + inputHeight + fixBelowInput + this.contentPadding.bottom ); float maxContainerHeight = boxConstraints.maxHeight - bottomHeight; float containerHeight = this.expands ? maxContainerHeight : Math.Min(contentHeight, maxContainerHeight); // Always position the prefix/suffix in the same place (baseline). float overflow = Math.Max(0, contentHeight - maxContainerHeight); float baselineAdjustment = fixAboveInput - overflow; // The baselines that will be used to draw the actual input text content. float inputBaseline = this.contentPadding.top + topHeight + inputInternalBaseline + baselineAdjustment; // The text in the input when an outline border is present is centered // within the container less 2.0 dps at the top to account for the vertical // space occupied by the floating label. float outlineBaseline = inputInternalBaseline + baselineAdjustment / 2 + (containerHeight - (2.0f + inputHeight)) / 2.0f; // Find the positions of the text below the input when it exists. float subtextCounterBaseline = 0; float subtextHelperBaseline = 0; float subtextCounterHeight = 0; float subtextHelperHeight = 0; if (this.counter != null) { subtextCounterBaseline = containerHeight + subtextGap + boxToBaseline.getOrDefault(this.counter, 0.0f); subtextCounterHeight = this.counter.size.height + subtextGap; } if (helperErrorExists) { subtextHelperBaseline = containerHeight + subtextGap + boxToBaseline.getOrDefault(this.helperError, 0.0f); subtextHelperHeight = helperErrorHeight; } float subtextBaseline = Math.Max( subtextCounterBaseline, subtextHelperBaseline ); float subtextHeight = Math.Max( subtextCounterHeight, subtextHelperHeight ); return new _RenderDecorationLayout( boxToBaseline: boxToBaseline, containerHeight: containerHeight, inputBaseline: inputBaseline, outlineBaseline: outlineBaseline, subtextBaseline: subtextBaseline, subtextHeight: subtextHeight ); } protected override float computeMinIntrinsicWidth(float height) { return _minWidth(this.icon, height) + this.contentPadding.left + _minWidth(this.prefixIcon, height) + _minWidth(this.prefix, height) + Mathf.Max(_minWidth(this.input, height), _minWidth(this.hint, height)) + _minWidth(this.suffix, height) + _minWidth(this.suffixIcon, height) + this.contentPadding.right; } protected override float computeMaxIntrinsicWidth(float height) { return _maxWidth(this.icon, height) + this.contentPadding.left + _maxWidth(this.prefixIcon, height) + _maxWidth(this.prefix, height) + Mathf.Max(_maxWidth(this.input, height), _maxWidth(this.hint, height)) + _maxWidth(this.suffix, height) + _maxWidth(this.suffixIcon, height) + this.contentPadding.right; } float _lineHeight(float width, List boxes) { float height = 0.0f; foreach (RenderBox box in boxes) { if (box == null) { continue; } height = Mathf.Max(_minHeight(box, width), height); } return height; } protected override float computeMinIntrinsicHeight(float width) { float subtextHeight = this._lineHeight(width, new List {this.helperError, this.counter}); if (subtextHeight > 0.0f) { subtextHeight += subtextGap; } return this.contentPadding.top + (this.label == null ? 0.0f : this.decoration.floatingLabelHeight) + this._lineHeight(width, new List {this.prefix, this.input, this.suffix}) + subtextHeight + this.contentPadding.bottom; } protected internal override float computeMaxIntrinsicHeight(float width) { return this.computeMinIntrinsicHeight(width); } protected override float? computeDistanceToActualBaseline(TextBaseline baseline) { return _boxParentData(this.input).offset.dy + this.input.getDistanceToActualBaseline(baseline); } Matrix4 _labelTransform; protected override void performLayout() { this._labelTransform = null; _RenderDecorationLayout layout = this._layout(this.constraints); float overallWidth = this.constraints.maxWidth; float? overallHeight = layout.containerHeight + layout.subtextHeight; if (this.container != null) { BoxConstraints containerConstraints = BoxConstraints.tightFor( height: layout.containerHeight, width: overallWidth - _boxSize(this.icon).width ); this.container.layout(containerConstraints, parentUsesSize: true); float x = _boxSize(this.icon).width; _boxParentData(this.container).offset = new Offset(x, 0.0f); } float height; float centerLayout(RenderBox box, float x) { _boxParentData(box).offset = new Offset(x, (height - box.size.height) / 2.0f); return box.size.width; } float baseline; float baselineLayout(RenderBox box, float x) { _boxParentData(box).offset = new Offset(x, baseline - layout.boxToBaseline[box]); return box.size.width; } float left = this.contentPadding.left; float right = overallWidth - this.contentPadding.right; height = layout.containerHeight ?? 0.0f; baseline = (this.decoration.isCollapsed || !this.decoration.border.isOutline ? layout.inputBaseline : layout.outlineBaseline) ?? 0.0f; if (this.icon != null) { float x = 0.0f; centerLayout(this.icon, x); } float start = left + _boxSize(this.icon).width; float end = right; if (this.prefixIcon != null) { start -= this.contentPadding.left; start += centerLayout(this.prefixIcon, start); } if (this.label != null) { if (this.decoration.alignLabelWithHint == true) { baselineLayout(this.label, start); } else { centerLayout(this.label, start); } } if (this.prefix != null) { start += baselineLayout(this.prefix, start); } if (this.input != null) { baselineLayout(this.input, start); } if (this.hint != null) { baselineLayout(this.hint, start); } if (this.suffixIcon != null) { end += this.contentPadding.right; end -= centerLayout(this.suffixIcon, end - this.suffixIcon.size.width); } if (this.suffix != null) { end -= baselineLayout(this.suffix, end - this.suffix.size.width); } if (this.helperError != null || this.counter != null) { height = layout.subtextHeight ?? 0.0f; baseline = layout.subtextBaseline ?? 0.0f; if (this.helperError != null) { baselineLayout(this.helperError, left + _boxSize(this.icon).width); } if (this.counter != null) { baselineLayout(this.counter, right - this.counter.size.width); } } if (this.label != null) { float labelX = _boxParentData(this.label).offset.dx; this.decoration.borderGap.start = labelX - _boxSize(this.icon).width; this.decoration.borderGap.extent = this.label.size.width * 0.75f; } else { this.decoration.borderGap.start = 0.0f; this.decoration.borderGap.extent = 0.0f; } this.size = this.constraints.constrain(new Size(overallWidth, overallHeight ?? 0.0f)); D.assert(this.size.width == this.constraints.constrainWidth(overallWidth)); D.assert(this.size.height == this.constraints.constrainHeight(overallHeight ?? 0.0f)); } void _paintLabel(PaintingContext context, Offset offset) { context.paintChild(this.label, offset); } public override void paint(PaintingContext context, Offset offset) { void doPaint(RenderBox child) { if (child != null) { context.paintChild(child, _boxParentData(child).offset + offset); } } doPaint(this.container); if (this.label != null) { Offset labelOffset = _boxParentData(this.label).offset; float labelHeight = this.label.size.height; float t = this.decoration.floatingLabelProgress; bool isOutlineBorder = this.decoration.border != null && this.decoration.border.isOutline; float floatingY = isOutlineBorder ? -labelHeight * 0.25f : this.contentPadding.top; float scale = MathUtils.lerpFloat(1.0f, 0.75f, t); float dx = labelOffset.dx; float dy = MathUtils.lerpFloat(0.0f, floatingY - labelOffset.dy, t); this._labelTransform = new Matrix4().identity(); this._labelTransform.translate(dx, labelOffset.dy + dy); this._labelTransform.scale(scale, scale, 1); context.pushTransform(this.needsCompositing, offset, this._labelTransform, this._paintLabel); } doPaint(this.icon); doPaint(this.prefix); doPaint(this.suffix); doPaint(this.prefixIcon); doPaint(this.suffixIcon); doPaint(this.hint); doPaint(this.input); doPaint(this.helperError); doPaint(this.counter); } protected override bool hitTestSelf(Offset position) { return true; } protected override bool hitTestChildren(BoxHitTestResult result, Offset position) { D.assert(position != null); foreach (RenderBox child in this._children) { Offset offset = _boxParentData(child).offset; bool isHit = result.addWithPaintOffset( offset: offset, position: position, hitTest: (BoxHitTestResult resultIn, Offset transformed) => { D.assert(transformed == position - offset); return child.hitTest(resultIn, position: transformed); } ); if (isHit) { return true; } } return false; } public override void applyPaintTransform(RenderObject child, Matrix4 transform) { if (child == this.label && this._labelTransform != null) { Offset labelOffset = _boxParentData(this.label).offset; transform.multiply(this._labelTransform); transform.translate(-labelOffset.dx, -labelOffset.dy); } base.applyPaintTransform(child, transform); } } class _RenderDecorationElement : RenderObjectElement { public _RenderDecorationElement(_Decorator widget) : base(widget) { } Dictionary<_DecorationSlot, Element> slotToChild = new Dictionary<_DecorationSlot, Element>(); Dictionary childToSlot = new Dictionary(); public new _Decorator widget { get { return (_Decorator) base.widget; } } public new _RenderDecoration renderObject { get { return (_RenderDecoration) base.renderObject; } } public override void visitChildren(ElementVisitor visitor) { this.slotToChild.Values.Each((child) => { visitor(child); }); } protected override void forgetChild(Element child) { D.assert(this.slotToChild.ContainsValue(child)); D.assert(this.childToSlot.ContainsKey(child)); _DecorationSlot slot = this.childToSlot[child]; this.childToSlot.Remove(child); this.slotToChild.Remove(slot); } void _mountChild(Widget widget, _DecorationSlot slot) { Element oldChild = this.slotToChild.getOrDefault(slot); Element newChild = this.updateChild(oldChild, widget, slot); if (oldChild != null) { this.slotToChild.Remove(slot); this.childToSlot.Remove(oldChild); } if (newChild != null) { this.slotToChild[slot] = newChild; this.childToSlot[newChild] = slot; } } public override void mount(Element parent, object newSlot) { base.mount(parent, newSlot); this._mountChild(this.widget.decoration.icon, _DecorationSlot.icon); this._mountChild(this.widget.decoration.input, _DecorationSlot.input); this._mountChild(this.widget.decoration.label, _DecorationSlot.label); this._mountChild(this.widget.decoration.hint, _DecorationSlot.hint); this._mountChild(this.widget.decoration.prefix, _DecorationSlot.prefix); this._mountChild(this.widget.decoration.suffix, _DecorationSlot.suffix); this._mountChild(this.widget.decoration.prefixIcon, _DecorationSlot.prefixIcon); this._mountChild(this.widget.decoration.suffixIcon, _DecorationSlot.suffixIcon); this._mountChild(this.widget.decoration.helperError, _DecorationSlot.helperError); this._mountChild(this.widget.decoration.counter, _DecorationSlot.counter); this._mountChild(this.widget.decoration.container, _DecorationSlot.container); } void _updateChild(Widget widget, _DecorationSlot slot) { Element oldChild = this.slotToChild.getOrDefault(slot); Element newChild = this.updateChild(oldChild, widget, slot); if (oldChild != null) { this.childToSlot.Remove(oldChild); this.slotToChild.Remove(slot); } if (newChild != null) { this.slotToChild[slot] = newChild; this.childToSlot[newChild] = slot; } } public override void update(Widget newWidget) { base.update(newWidget); D.assert(this.widget == newWidget); this._updateChild(this.widget.decoration.icon, _DecorationSlot.icon); this._updateChild(this.widget.decoration.input, _DecorationSlot.input); this._updateChild(this.widget.decoration.label, _DecorationSlot.label); this._updateChild(this.widget.decoration.hint, _DecorationSlot.hint); this._updateChild(this.widget.decoration.prefix, _DecorationSlot.prefix); this._updateChild(this.widget.decoration.suffix, _DecorationSlot.suffix); this._updateChild(this.widget.decoration.prefixIcon, _DecorationSlot.prefixIcon); this._updateChild(this.widget.decoration.suffixIcon, _DecorationSlot.suffixIcon); this._updateChild(this.widget.decoration.helperError, _DecorationSlot.helperError); this._updateChild(this.widget.decoration.counter, _DecorationSlot.counter); this._updateChild(this.widget.decoration.container, _DecorationSlot.container); } void _updateRenderObject(RenderObject child, _DecorationSlot slot) { switch (slot) { case _DecorationSlot.icon: this.renderObject.icon = (RenderBox) child; break; case _DecorationSlot.input: this.renderObject.input = (RenderBox) child; break; case _DecorationSlot.label: this.renderObject.label = (RenderBox) child; break; case _DecorationSlot.hint: this.renderObject.hint = (RenderBox) child; break; case _DecorationSlot.prefix: this.renderObject.prefix = (RenderBox) child; break; case _DecorationSlot.suffix: this.renderObject.suffix = (RenderBox) child; break; case _DecorationSlot.prefixIcon: this.renderObject.prefixIcon = (RenderBox) child; break; case _DecorationSlot.suffixIcon: this.renderObject.suffixIcon = (RenderBox) child; break; case _DecorationSlot.helperError: this.renderObject.helperError = (RenderBox) child; break; case _DecorationSlot.counter: this.renderObject.counter = (RenderBox) child; break; case _DecorationSlot.container: this.renderObject.container = (RenderBox) child; break; } } protected override void insertChildRenderObject(RenderObject child, object slotValue) { D.assert(child is RenderBox); D.assert(slotValue is _DecorationSlot); _DecorationSlot slot = (_DecorationSlot) slotValue; this._updateRenderObject(child, slot); D.assert(this.renderObject.childToSlot.ContainsKey((RenderBox) child)); D.assert(this.renderObject.slotToChild.ContainsKey(slot)); } protected override void removeChildRenderObject(RenderObject child) { D.assert(child is RenderBox); D.assert(this.renderObject.childToSlot.ContainsKey((RenderBox) child)); var slot = this.renderObject.childToSlot[(RenderBox) child]; this._updateRenderObject(null, this.renderObject.childToSlot[(RenderBox) child]); D.assert(!this.renderObject.childToSlot.ContainsKey((RenderBox) child)); D.assert(!this.renderObject.slotToChild.ContainsKey(slot)); } protected override void moveChildRenderObject(RenderObject child, object slotValue) { D.assert(false, () => "not reachable"); } } class _Decorator : RenderObjectWidget { public _Decorator( Key key = null, _Decoration decoration = null, TextBaseline? textBaseline = null, bool isFocused = false, bool? expands = null ) : base(key: key) { D.assert(decoration != null); D.assert(textBaseline != null); D.assert(expands != null); this.decoration = decoration; this.textBaseline = textBaseline; this.isFocused = isFocused; this.expands = expands.Value; } public readonly _Decoration decoration; public readonly TextBaseline? textBaseline; public readonly bool isFocused; public readonly bool expands; public override Element createElement() { return new _RenderDecorationElement(this); } public override RenderObject createRenderObject(BuildContext context) { return new _RenderDecoration( decoration: this.decoration, textBaseline: this.textBaseline, isFocused: this.isFocused, expands: this.expands ); } public override void updateRenderObject(BuildContext context, RenderObject _renderObject) { _RenderDecoration renderObject = _renderObject as _RenderDecoration; renderObject.decoration = this.decoration; renderObject.textBaseline = this.textBaseline; renderObject.isFocused = this.isFocused; renderObject.expands = this.expands; } } class _AffixText : StatelessWidget { public _AffixText( bool labelIsFloating = false, string text = null, TextStyle style = null, Widget child = null ) { this.labelIsFloating = labelIsFloating; this.text = text; this.style = style; this.child = child; } public readonly bool labelIsFloating; public readonly string text; public readonly TextStyle style; public readonly Widget child; public override Widget build(BuildContext context) { return DefaultTextStyle.merge( style: this.style, child: new AnimatedOpacity( duration: InputDecoratorConstants._kTransitionDuration, curve: InputDecoratorConstants._kTransitionCurve, opacity: this.labelIsFloating ? 1.0f : 0.0f, child: this.child ?? new Text(this.text, style: this.style) ) ); } } public class InputDecorator : StatefulWidget { public InputDecorator( Key key = null, InputDecoration decoration = null, TextStyle baseStyle = null, TextAlign? textAlign = null, bool isFocused = false, bool expands = false, bool isEmpty = false, Widget child = null ) : base(key: key) { this.decoration = decoration; this.baseStyle = baseStyle; this.textAlign = textAlign; this.isFocused = isFocused; this.expands = expands; this.isEmpty = isEmpty; this.child = child; } public readonly InputDecoration decoration; public readonly TextStyle baseStyle; public readonly TextAlign? textAlign; public readonly bool isFocused; public readonly bool expands; public readonly bool isEmpty; public readonly Widget child; public bool _labelShouldWithdraw { get { return !this.isEmpty || this.isFocused; } } public override State createState() { return new _InputDecoratorState(); } internal static RenderBox containerOf(BuildContext context) { _RenderDecoration result = (_RenderDecoration) context.ancestorRenderObjectOfType(new TypeMatcher<_RenderDecoration>()); return result?.container; } public override void debugFillProperties(DiagnosticPropertiesBuilder properties) { base.debugFillProperties(properties); properties.add(new DiagnosticsProperty("decoration", this.decoration)); properties.add(new DiagnosticsProperty("baseStyle", this.baseStyle, defaultValue: null)); properties.add(new DiagnosticsProperty("isFocused", this.isFocused)); properties.add(new DiagnosticsProperty("expands", this.expands)); properties.add(new DiagnosticsProperty("isEmpty", this.isEmpty)); } } class _InputDecoratorState : TickerProviderStateMixin { AnimationController _floatingLabelController; AnimationController _shakingLabelController; _InputBorderGap _borderGap = new _InputBorderGap(); public override void initState() { base.initState(); this._floatingLabelController = new AnimationController( duration: InputDecoratorConstants._kTransitionDuration, vsync: this, value: (this.widget.decoration.hasFloatingPlaceholder == true && this.widget._labelShouldWithdraw) ? 1.0f : 0.0f ); this._floatingLabelController.addListener(this._handleChange); this._shakingLabelController = new AnimationController( duration: InputDecoratorConstants._kTransitionDuration, vsync: this ); } public override void didChangeDependencies() { base.didChangeDependencies(); this._effectiveDecoration = null; } public override void dispose() { this._floatingLabelController.dispose(); this._shakingLabelController.dispose(); base.dispose(); } void _handleChange() { this.setState(() => { }); } InputDecoration _effectiveDecoration; public InputDecoration decoration { get { this._effectiveDecoration = this._effectiveDecoration ?? this.widget.decoration.applyDefaults( Theme.of(this.context).inputDecorationTheme ); return this._effectiveDecoration; } } TextAlign? textAlign { get { return this.widget.textAlign; } } bool isFocused { get { return this.widget.isFocused; } } bool isEmpty { get { return this.widget.isEmpty; } } public override void didUpdateWidget(StatefulWidget _old) { base.didUpdateWidget(_old); InputDecorator old = _old as InputDecorator; if (this.widget.decoration != old.decoration) { this._effectiveDecoration = null; } if (this.widget._labelShouldWithdraw != old._labelShouldWithdraw && this.widget.decoration.hasFloatingPlaceholder == true) { if (this.widget._labelShouldWithdraw) { this._floatingLabelController.forward(); } else { this._floatingLabelController.reverse(); } } string errorText = this.decoration.errorText; string oldErrorText = old.decoration.errorText; if (this._floatingLabelController.isCompleted && errorText != null && errorText != oldErrorText) { this._shakingLabelController.setValue(0.0f); this._shakingLabelController.forward(); } } Color _getActiveColor(ThemeData themeData) { if (this.isFocused) { switch (themeData.brightness) { case Brightness.dark: return themeData.accentColor; case Brightness.light: return themeData.primaryColor; } } return themeData.hintColor; } Color _getFillColor(ThemeData themeData) { if (this.decoration.filled != true) { return Colors.transparent; } if (this.decoration.fillColor != null) { return this.decoration.fillColor; } Color darkEnabled = new Color(0x1AFFFFFF); Color darkDisabled = new Color(0x0DFFFFFF); Color lightEnabled = new Color(0x0A000000); Color lightDisabled = new Color(0x05000000); switch (themeData.brightness) { case Brightness.dark: return this.decoration.enabled == true ? darkEnabled : darkDisabled; case Brightness.light: return this.decoration.enabled == true ? lightEnabled : lightDisabled; } return lightEnabled; } Color _getDefaultIconColor(ThemeData themeData) { if (!this.decoration.enabled == true) { return themeData.disabledColor; } switch (themeData.brightness) { case Brightness.dark: return Colors.white70; case Brightness.light: return Colors.black45; default: return themeData.iconTheme.color; } } bool _hasInlineLabel { get { return !this.widget._labelShouldWithdraw && this.decoration.labelText != null; } } bool _shouldShowLabel { get { return this._hasInlineLabel || this.decoration.hasFloatingPlaceholder == true; } } TextStyle _getInlineStyle(ThemeData themeData) { return themeData.textTheme.subhead.merge(this.widget.baseStyle) .copyWith(color: this.decoration.enabled == true ? themeData.hintColor : themeData.disabledColor); } TextStyle _getFloatingLabelStyle(ThemeData themeData) { Color color = this.decoration.errorText != null ? this.decoration.errorStyle?.color ?? themeData.errorColor : this._getActiveColor(themeData); TextStyle style = themeData.textTheme.subhead.merge(this.widget.baseStyle); return style .copyWith(color: this.decoration.enabled == true ? color : themeData.disabledColor) .merge(this.decoration.labelStyle); } TextStyle _getHelperStyle(ThemeData themeData) { Color color = this.decoration.enabled == true ? themeData.hintColor : Colors.transparent; return themeData.textTheme.caption.copyWith(color: color).merge(this.decoration.helperStyle); } TextStyle _getErrorStyle(ThemeData themeData) { Color color = this.decoration.enabled == true ? themeData.errorColor : Colors.transparent; return themeData.textTheme.caption.copyWith(color: color).merge(this.decoration.errorStyle); } InputBorder _getDefaultBorder(ThemeData themeData) { if (this.decoration.border?.borderSide == BorderSide.none) { return this.decoration.border; } Color borderColor; if (this.decoration.enabled == true) { borderColor = this.decoration.errorText == null ? this._getActiveColor(themeData) : themeData.errorColor; } else { borderColor = (this.decoration.filled == true && this.decoration.border?.isOutline != true) ? Colors.transparent : themeData.disabledColor; } float borderWeight; if (this.decoration.isCollapsed || this.decoration?.border == InputBorder.none || !this.decoration.enabled == true) { borderWeight = 0.0f; } else { borderWeight = this.isFocused ? 2.0f : 1.0f; } InputBorder border = this.decoration.border ?? new UnderlineInputBorder(); return border.copyWith(borderSide: new BorderSide(color: borderColor, width: borderWeight)); } public override Widget build(BuildContext context) { ThemeData themeData = Theme.of(context); TextStyle inlineStyle = this._getInlineStyle(themeData); TextBaseline? textBaseline = inlineStyle.textBaseline; TextStyle hintStyle = inlineStyle.merge(this.decoration.hintStyle); Widget hint = this.decoration.hintText == null ? null : new AnimatedOpacity( opacity: (this.isEmpty && !this._hasInlineLabel) ? 1.0f : 0.0f, duration: InputDecoratorConstants._kTransitionDuration, curve: InputDecoratorConstants._kTransitionCurve, child: new Text(this.decoration.hintText, style: hintStyle, overflow: TextOverflow.ellipsis, textAlign: this.textAlign, maxLines: this.decoration.hintMaxLines ) ); bool isError = this.decoration.errorText != null; InputBorder border; if (!this.decoration.enabled == true) { border = isError ? this.decoration.errorBorder : this.decoration.disabledBorder; } else if (this.isFocused) { border = isError ? this.decoration.focusedErrorBorder : this.decoration.focusedBorder; } else { border = isError ? this.decoration.errorBorder : this.decoration.enabledBorder; } border = border ?? this._getDefaultBorder(themeData); Widget container = new _BorderContainer( border: border, gap: this._borderGap, gapAnimation: this._floatingLabelController.view, fillColor: this._getFillColor(themeData) ); TextStyle inlineLabelStyle = inlineStyle.merge(this.decoration.labelStyle); Widget label = this.decoration.labelText == null ? null : new _Shaker( animation: this._shakingLabelController.view, child: new AnimatedOpacity( duration: InputDecoratorConstants._kTransitionDuration, curve: InputDecoratorConstants._kTransitionCurve, opacity: this._shouldShowLabel ? 1.0f : 0.0f, child: new AnimatedDefaultTextStyle( duration: InputDecoratorConstants._kTransitionDuration, curve: InputDecoratorConstants._kTransitionCurve, style: this.widget._labelShouldWithdraw ? this._getFloatingLabelStyle(themeData) : inlineLabelStyle, child: new Text(this.decoration.labelText, overflow: TextOverflow.ellipsis, textAlign: this.textAlign ) ) ) ); Widget prefix = this.decoration.prefix == null && this.decoration.prefixText == null ? null : new _AffixText( labelIsFloating: this.widget._labelShouldWithdraw, text: this.decoration.prefixText, style: this.decoration.prefixStyle ?? hintStyle, child: this.decoration.prefix ); Widget suffix = this.decoration.suffix == null && this.decoration.suffixText == null ? null : new _AffixText( labelIsFloating: this.widget._labelShouldWithdraw, text: this.decoration.suffixText, style: this.decoration.suffixStyle ?? hintStyle, child: this.decoration.suffix ); Color activeColor = this._getActiveColor(themeData); bool decorationIsDense = this.decoration.isDense == true; float iconSize = decorationIsDense ? 18.0f : 24.0f; Color iconColor = this.isFocused ? activeColor : this._getDefaultIconColor(themeData); Widget icon = this.decoration.icon == null ? null : new Padding( padding: EdgeInsets.only(right: 16.0f), child: IconTheme.merge( data: new IconThemeData( color: iconColor, size: iconSize ), child: this.decoration.icon ) ); Widget prefixIcon = this.decoration.prefixIcon == null ? null : new Center( widthFactor: 1.0f, heightFactor: 1.0f, child: new ConstrainedBox( constraints: new BoxConstraints(minWidth: 48.0f, minHeight: 48.0f), child: IconTheme.merge( data: new IconThemeData( color: iconColor, size: iconSize ), child: this.decoration.prefixIcon ) ) ); Widget suffixIcon = this.decoration.suffixIcon == null ? null : new Center( widthFactor: 1.0f, heightFactor: 1.0f, child: new ConstrainedBox( constraints: new BoxConstraints(minWidth: 48.0f, minHeight: 48.0f), child: IconTheme.merge( data: new IconThemeData( color: iconColor, size: iconSize ), child: this.decoration.suffixIcon ) ) ); Widget helperError = new _HelperError( textAlign: this.textAlign, helperText: this.decoration.helperText, helperStyle: this._getHelperStyle(themeData), errorText: this.decoration.errorText, errorStyle: this._getErrorStyle(themeData), errorMaxLines: this.decoration.errorMaxLines ); Widget counter = null; if (this.decoration.counter != null) { counter = this.decoration.counter; } else if (this.decoration.counterText != null && this.decoration.counterText != "") { counter = new Text(this.decoration.counterText, style: this._getHelperStyle(themeData).merge(this.decoration.counterStyle), overflow: TextOverflow.ellipsis ); } EdgeInsets decorationContentPadding = this.decoration.contentPadding; EdgeInsets contentPadding; float? floatingLabelHeight; if (this.decoration.isCollapsed) { floatingLabelHeight = 0.0f; contentPadding = decorationContentPadding ?? EdgeInsets.zero; } else if (!border.isOutline) { floatingLabelHeight = (4.0f + 0.75f * inlineLabelStyle.fontSize) * MediaQuery.textScaleFactorOf(context); if (this.decoration.filled == true) { contentPadding = decorationContentPadding ?? (decorationIsDense ? EdgeInsets.fromLTRB(12.0f, 8.0f, 12.0f, 8.0f) : EdgeInsets.fromLTRB(12.0f, 12.0f, 12.0f, 12.0f)); } else { contentPadding = decorationContentPadding ?? (decorationIsDense ? EdgeInsets.fromLTRB(0.0f, 8.0f, 0.0f, 8.0f) : EdgeInsets.fromLTRB(0.0f, 12.0f, 0.0f, 12.0f)); } } else { floatingLabelHeight = 0.0f; contentPadding = decorationContentPadding ?? (decorationIsDense ? EdgeInsets.fromLTRB(12.0f, 20.0f, 12.0f, 12.0f) : EdgeInsets.fromLTRB(12.0f, 24.0f, 12.0f, 16.0f)); } return new _Decorator( decoration: new _Decoration( contentPadding: contentPadding, isCollapsed: this.decoration.isCollapsed, floatingLabelHeight: floatingLabelHeight ?? 0.0f, floatingLabelProgress: this._floatingLabelController.value, border: border, borderGap: this._borderGap, icon: icon, input: this.widget.child, label: label, alignLabelWithHint: this.decoration.alignLabelWithHint, hint: hint, prefix: prefix, suffix: suffix, prefixIcon: prefixIcon, suffixIcon: suffixIcon, helperError: helperError, counter: counter, container: container ), textBaseline: textBaseline, isFocused: this.isFocused, expands: this.widget.expands ); } } public class InputDecoration { public InputDecoration( Widget icon = null, string labelText = null, TextStyle labelStyle = null, string helperText = null, TextStyle helperStyle = null, string hintText = null, TextStyle hintStyle = null, int? hintMaxLines = null, string errorText = null, TextStyle errorStyle = null, int? errorMaxLines = null, bool? hasFloatingPlaceholder = true, bool? isDense = null, EdgeInsets contentPadding = null, Widget prefixIcon = null, Widget prefix = null, string prefixText = null, TextStyle prefixStyle = null, Widget suffixIcon = null, Widget suffix = null, string suffixText = null, TextStyle suffixStyle = null, Widget counter = null, string counterText = null, TextStyle counterStyle = null, bool? filled = null, Color fillColor = null, InputBorder errorBorder = null, InputBorder focusedBorder = null, InputBorder focusedErrorBorder = null, InputBorder disabledBorder = null, InputBorder enabledBorder = null, InputBorder border = null, bool? enabled = true, bool? alignLabelWithHint = null ) { D.assert(enabled != null); D.assert(!(prefix != null && prefixText != null), () => "Declaring both prefix and prefixText is not supported"); D.assert(!(suffix != null && suffixText != null), () => "Declaring both suffix and suffixText is not supported"); this.isCollapsed = false; this.icon = icon; this.labelText = labelText; this.labelStyle = labelStyle; this.helperText = helperText; this.helperStyle = helperStyle; this.hintText = hintText; this.hintStyle = hintStyle; this.hintMaxLines = hintMaxLines; this.errorText = errorText; this.errorStyle = errorStyle; this.errorMaxLines = errorMaxLines; this.hasFloatingPlaceholder = hasFloatingPlaceholder; this.isDense = isDense; this.contentPadding = contentPadding; this.prefix = prefix; this.prefixText = prefixText; this.prefixIcon = prefixIcon; this.prefixStyle = prefixStyle; this.suffix = suffix; this.suffixText = suffixText; this.suffixIcon = suffixIcon; this.suffixStyle = suffixStyle; this.counter = counter; this.counterText = counterText; this.counterStyle = counterStyle; this.filled = filled; this.fillColor = fillColor; this.errorBorder = errorBorder; this.focusedBorder = focusedBorder; this.focusedErrorBorder = focusedErrorBorder; this.disabledBorder = disabledBorder; this.enabledBorder = enabledBorder; this.border = border; this.enabled = enabled; this.alignLabelWithHint = alignLabelWithHint; } public static InputDecoration collapsed( string hintText = null, bool hasFloatingPlaceholder = true, TextStyle hintStyle = null, bool filled = false, Color fillColor = null, InputBorder border = null, bool enabled = true ) { border = border ?? InputBorder.none; InputDecoration decoration = new InputDecoration( icon: null, labelText: null, labelStyle: null, helperText: null, helperStyle: null, hintMaxLines: null, errorText: null, errorStyle: null, errorMaxLines: null, isDense: false, contentPadding: EdgeInsets.zero, prefixIcon: null, prefix: null, prefixText: null, prefixStyle: null, suffix: null, suffixIcon: null, suffixText: null, suffixStyle: null, counter: null, counterText: null, counterStyle: null, errorBorder: null, focusedBorder: null, focusedErrorBorder: null, disabledBorder: null, enabledBorder: null, hintText: hintText, hasFloatingPlaceholder: hasFloatingPlaceholder, hintStyle: hintStyle, filled: filled, fillColor: fillColor, border: border, enabled: enabled, alignLabelWithHint: false ); decoration.isCollapsed = true; return decoration; } public readonly Widget icon; public readonly string labelText; public readonly TextStyle labelStyle; public readonly string helperText; public readonly TextStyle helperStyle; public readonly string hintText; public readonly TextStyle hintStyle; public readonly int? hintMaxLines; public readonly string errorText; public readonly TextStyle errorStyle; public readonly int? errorMaxLines; public readonly bool? hasFloatingPlaceholder; public readonly bool? isDense; public readonly EdgeInsets contentPadding; public bool isCollapsed; public readonly Widget prefixIcon; public readonly Widget prefix; public readonly string prefixText; public readonly TextStyle prefixStyle; public readonly Widget suffixIcon; public readonly Widget suffix; public readonly string suffixText; public readonly TextStyle suffixStyle; public readonly Widget counter; public readonly string counterText; public readonly TextStyle counterStyle; public readonly bool? filled; public readonly Color fillColor; public readonly InputBorder errorBorder; public readonly InputBorder focusedBorder; public readonly InputBorder focusedErrorBorder; public readonly InputBorder disabledBorder; public readonly InputBorder enabledBorder; public readonly InputBorder border; public readonly bool? enabled; public readonly bool? alignLabelWithHint; public InputDecoration copyWith( Widget icon = null, string labelText = null, TextStyle labelStyle = null, string helperText = null, TextStyle helperStyle = null, string hintText = null, TextStyle hintStyle = null, int? hintMaxLines = null, string errorText = null, TextStyle errorStyle = null, int? errorMaxLines = null, bool? hasFloatingPlaceholder = null, bool? isDense = null, EdgeInsets contentPadding = null, Widget prefixIcon = null, Widget prefix = null, string prefixText = null, TextStyle prefixStyle = null, Widget suffixIcon = null, Widget suffix = null, string suffixText = null, TextStyle suffixStyle = null, Widget counter = null, string counterText = null, TextStyle counterStyle = null, bool? filled = null, Color fillColor = null, InputBorder errorBorder = null, InputBorder focusedBorder = null, InputBorder focusedErrorBorder = null, InputBorder disabledBorder = null, InputBorder enabledBorder = null, InputBorder border = null, bool? enabled = null, bool? alignLabelWithHint = null ) { return new InputDecoration( icon: icon ?? this.icon, labelText: labelText ?? this.labelText, labelStyle: labelStyle ?? this.labelStyle, helperText: helperText ?? this.helperText, helperStyle: helperStyle ?? this.helperStyle, hintText: hintText ?? this.hintText, hintStyle: hintStyle ?? this.hintStyle, hintMaxLines: hintMaxLines ?? this.hintMaxLines, errorText: errorText ?? this.errorText, errorStyle: errorStyle ?? this.errorStyle, errorMaxLines: errorMaxLines ?? this.errorMaxLines, hasFloatingPlaceholder: hasFloatingPlaceholder ?? this.hasFloatingPlaceholder, isDense: isDense ?? this.isDense, contentPadding: contentPadding ?? this.contentPadding, prefixIcon: prefixIcon ?? this.prefixIcon, prefix: prefix ?? this.prefix, prefixText: prefixText ?? this.prefixText, prefixStyle: prefixStyle ?? this.prefixStyle, suffixIcon: suffixIcon ?? this.suffixIcon, suffix: suffix ?? this.suffix, suffixText: suffixText ?? this.suffixText, suffixStyle: suffixStyle ?? this.suffixStyle, counter: counter ?? this.counter, counterText: counterText ?? this.counterText, counterStyle: counterStyle ?? this.counterStyle, filled: filled ?? this.filled, fillColor: fillColor ?? this.fillColor, errorBorder: errorBorder ?? this.errorBorder, focusedBorder: focusedBorder ?? this.focusedBorder, focusedErrorBorder: focusedErrorBorder ?? this.focusedErrorBorder, disabledBorder: disabledBorder ?? this.disabledBorder, enabledBorder: enabledBorder ?? this.enabledBorder, border: border ?? this.border, enabled: enabled ?? this.enabled, alignLabelWithHint: alignLabelWithHint ?? this.alignLabelWithHint ); } public InputDecoration applyDefaults(InputDecorationTheme theme) { return this.copyWith( labelStyle: this.labelStyle ?? theme.labelStyle, helperStyle: this.helperStyle ?? theme.helperStyle, hintStyle: this.hintStyle ?? theme.hintStyle, errorStyle: this.errorStyle ?? theme.errorStyle, errorMaxLines: this.errorMaxLines ?? theme.errorMaxLines, hasFloatingPlaceholder: this.hasFloatingPlaceholder ?? theme.hasFloatingPlaceholder, isDense: this.isDense ?? theme.isDense, contentPadding: this.contentPadding ?? theme.contentPadding, prefixStyle: this.prefixStyle ?? theme.prefixStyle, suffixStyle: this.suffixStyle ?? theme.suffixStyle, counterStyle: this.counterStyle ?? theme.counterStyle, filled: this.filled ?? theme.filled, fillColor: this.fillColor ?? theme.fillColor, errorBorder: this.errorBorder ?? theme.errorBorder, focusedBorder: this.focusedBorder ?? theme.focusedBorder, focusedErrorBorder: this.focusedErrorBorder ?? theme.focusedErrorBorder, disabledBorder: this.disabledBorder ?? theme.disabledBorder, enabledBorder: this.enabledBorder ?? theme.enabledBorder, border: this.border ?? theme.border, alignLabelWithHint: this.alignLabelWithHint ?? theme.alignLabelWithHint ); } public static bool operator ==(InputDecoration left, InputDecoration right) { return Equals(left, right); } public static bool operator !=(InputDecoration left, InputDecoration right) { return !Equals(left, right); } public bool Equals(InputDecoration other) { return Equals(other.icon, this.icon) && Equals(other.labelText, this.labelText) && Equals(other.labelStyle, this.labelStyle) && Equals(other.helperText, this.helperText) && Equals(other.helperStyle, this.helperStyle) && Equals(other.hintText, this.hintText) && Equals(other.hintStyle, this.hintStyle) && Equals(other.hintMaxLines, this.hintMaxLines) && Equals(other.errorText, this.errorText) && Equals(other.errorStyle, this.errorStyle) && Equals(other.errorMaxLines, this.errorMaxLines) && Equals(other.hasFloatingPlaceholder, this.hasFloatingPlaceholder) && Equals(other.isDense, this.isDense) && Equals(other.contentPadding, this.contentPadding) && Equals(other.isCollapsed, this.isCollapsed) && Equals(other.prefixIcon, this.prefixIcon) && Equals(other.prefix, this.prefix) && Equals(other.prefixText, this.prefixText) && Equals(other.prefixStyle, this.prefixStyle) && Equals(other.suffixIcon, this.suffixIcon) && Equals(other.suffix, this.suffix) && Equals(other.suffixText, this.suffixText) && Equals(other.suffixStyle, this.suffixStyle) && Equals(other.counter, this.counter) && Equals(other.counterText, this.counterText) && Equals(other.counterStyle, this.counterStyle) && Equals(other.filled, this.filled) && Equals(other.fillColor, this.fillColor) && Equals(other.errorBorder, this.errorBorder) && Equals(other.focusedBorder, this.focusedBorder) && Equals(other.focusedErrorBorder, this.focusedErrorBorder) && Equals(other.disabledBorder, this.disabledBorder) && Equals(other.enabledBorder, this.enabledBorder) && Equals(other.border, this.border) && Equals(other.enabled, this.enabled) && Equals(other.alignLabelWithHint, this.alignLabelWithHint); } public override bool Equals(object obj) { if (ReferenceEquals(null, obj)) { return false; } if (ReferenceEquals(this, obj)) { return true; } if (obj.GetType() != this.GetType()) { return false; } return this.Equals((InputDecoration) obj); } public override int GetHashCode() { unchecked { var hashCode = this.icon.GetHashCode(); hashCode = (hashCode * 397) ^ this.labelText.GetHashCode(); hashCode = (hashCode * 397) ^ this.labelStyle.GetHashCode(); hashCode = (hashCode * 397) ^ this.helperText.GetHashCode(); hashCode = (hashCode * 397) ^ this.helperStyle.GetHashCode(); hashCode = (hashCode * 397) ^ this.hintText.GetHashCode(); hashCode = (hashCode * 397) ^ this.hintStyle.GetHashCode(); hashCode = (hashCode * 397) ^ this.hintMaxLines.GetHashCode(); hashCode = (hashCode * 397) ^ this.errorText.GetHashCode(); hashCode = (hashCode * 397) ^ this.errorStyle.GetHashCode(); hashCode = (hashCode * 397) ^ this.errorMaxLines.GetHashCode(); hashCode = (hashCode * 397) ^ this.hasFloatingPlaceholder.GetHashCode(); hashCode = (hashCode * 397) ^ this.isDense.GetHashCode(); hashCode = (hashCode * 397) ^ this.contentPadding.GetHashCode(); hashCode = (hashCode * 397) ^ this.isCollapsed.GetHashCode(); hashCode = (hashCode * 397) ^ this.filled.GetHashCode(); hashCode = (hashCode * 397) ^ this.fillColor.GetHashCode(); hashCode = (hashCode * 397) ^ this.border.GetHashCode(); hashCode = (hashCode * 397) ^ this.enabled.GetHashCode(); hashCode = (hashCode * 397) ^ this.prefixIcon.GetHashCode(); hashCode = (hashCode * 397) ^ this.prefix.GetHashCode(); hashCode = (hashCode * 397) ^ this.prefixText.GetHashCode(); hashCode = (hashCode * 397) ^ this.prefixStyle.GetHashCode(); hashCode = (hashCode * 397) ^ this.suffixIcon.GetHashCode(); hashCode = (hashCode * 397) ^ this.suffix.GetHashCode(); hashCode = (hashCode * 397) ^ this.suffixText.GetHashCode(); hashCode = (hashCode * 397) ^ this.suffixStyle.GetHashCode(); hashCode = (hashCode * 397) ^ this.counter.GetHashCode(); hashCode = (hashCode * 397) ^ this.counterText.GetHashCode(); hashCode = (hashCode * 397) ^ this.counterStyle.GetHashCode(); hashCode = (hashCode * 397) ^ this.filled.GetHashCode(); hashCode = (hashCode * 397) ^ this.fillColor.GetHashCode(); hashCode = (hashCode * 397) ^ this.errorBorder.GetHashCode(); hashCode = (hashCode * 397) ^ this.focusedBorder.GetHashCode(); hashCode = (hashCode * 397) ^ this.focusedErrorBorder.GetHashCode(); hashCode = (hashCode * 397) ^ this.disabledBorder.GetHashCode(); hashCode = (hashCode * 397) ^ this.enabledBorder.GetHashCode(); hashCode = (hashCode * 397) ^ this.border.GetHashCode(); hashCode = (hashCode * 397) ^ this.enabled.GetHashCode(); hashCode = (hashCode * 397) ^ this.alignLabelWithHint.GetHashCode(); return hashCode; } } public override string ToString() { List description = new List { }; if (this.icon != null) { description.Add($"icon: ${this.icon}"); } if (this.labelText != null) { description.Add($"labelText: ${this.labelText}"); } if (this.helperText != null) { description.Add($"helperText: ${this.helperText}"); } if (this.hintMaxLines != null) { description.Add($"hintMaxLines: ${this.hintMaxLines}"); } if (this.hintText != null) { description.Add($"hintText: ${this.hintText}"); } if (this.errorText != null) { description.Add($"errorText: ${this.errorText}"); } if (this.errorStyle != null) { description.Add($"errorStyle: ${this.errorStyle}"); } if (this.errorMaxLines != null) { description.Add($"errorMaxLines: ${this.errorMaxLines}"); } if (this.hasFloatingPlaceholder == false) { description.Add($"hasFloatingPlaceholder: false"); } if (this.isDense ?? false) { description.Add($"isDense: ${this.isDense}"); } if (this.contentPadding != null) { description.Add($"contentPadding: ${this.contentPadding}"); } if (this.isCollapsed) { description.Add($"isCollapsed: ${this.isCollapsed}"); } if (this.prefixIcon != null) { description.Add($"prefixIcon: ${this.prefixIcon}"); } if (this.prefix != null) { description.Add($"prefix: ${this.prefix}"); } if (this.prefixText != null) { description.Add($"prefixText: ${this.prefixText}"); } if (this.prefixStyle != null) { description.Add($"prefixStyle: ${this.prefixStyle}"); } if (this.suffixIcon != null) { description.Add($"suffixIcon: ${this.suffixIcon}"); } if (this.suffix != null) { description.Add($"suffix: ${this.suffix}"); } if (this.suffixText != null) { description.Add($"suffixText: ${this.suffixText}"); } if (this.suffixStyle != null) { description.Add($"suffixStyle: ${this.suffixStyle}"); } if (this.counter != null) { description.Add($"counter: ${this.counter}"); } if (this.counterText != null) { description.Add($"counterText: ${this.counterText}"); } if (this.counterStyle != null) { description.Add($"counterStyle: ${this.counterStyle}"); } if (this.filled == true) { description.Add($"filled: true"); } if (this.fillColor != null) { description.Add($"fillColor: ${this.fillColor}"); } if (this.errorBorder != null) { description.Add($"errorBorder: ${this.errorBorder}"); } if (this.focusedBorder != null) { description.Add($"focusedBorder: ${this.focusedBorder}"); } if (this.focusedErrorBorder != null) { description.Add($"focusedErrorBorder: ${this.focusedErrorBorder}"); } if (this.disabledBorder != null) { description.Add($"disabledBorder: ${this.disabledBorder}"); } if (this.enabledBorder != null) { description.Add($"enabledBorder: ${this.enabledBorder}"); } if (this.border != null) { description.Add($"border: ${this.border}"); } if (this.enabled != true) { description.Add("enabled: false"); } if (this.alignLabelWithHint != null) { description.Add($"alignLabelWithHint: {this.alignLabelWithHint}"); } return $"InputDecoration(${string.Join(", ", description)})"; } } public class InputDecorationTheme : Diagnosticable { public InputDecorationTheme( TextStyle labelStyle = null, TextStyle helperStyle = null, TextStyle hintStyle = null, TextStyle errorStyle = null, int? errorMaxLines = null, bool? hasFloatingPlaceholder = true, bool? isDense = false, EdgeInsets contentPadding = null, bool? isCollapsed = false, TextStyle prefixStyle = null, TextStyle suffixStyle = null, TextStyle counterStyle = null, bool? filled = false, Color fillColor = null, InputBorder errorBorder = null, InputBorder focusedBorder = null, InputBorder focusedErrorBorder = null, InputBorder disabledBorder = null, InputBorder enabledBorder = null, InputBorder border = null, bool alignLabelWithHint = false ) { D.assert(isDense != null); D.assert(isCollapsed != null); D.assert(filled != null); this.labelStyle = labelStyle; this.helperStyle = helperStyle; this.hintStyle = hintStyle; this.errorStyle = errorStyle; this.errorMaxLines = errorMaxLines; this.hasFloatingPlaceholder = hasFloatingPlaceholder; this.isDense = isDense; this.contentPadding = contentPadding; this.isCollapsed = isCollapsed; this.prefixStyle = prefixStyle; this.suffixStyle = suffixStyle; this.counterStyle = counterStyle; this.filled = filled; this.fillColor = fillColor; this.errorBorder = errorBorder; this.focusedBorder = focusedBorder; this.focusedErrorBorder = focusedErrorBorder; this.disabledBorder = disabledBorder; this.enabledBorder = enabledBorder; this.border = border; this.alignLabelWithHint = alignLabelWithHint; } public readonly TextStyle labelStyle; public readonly TextStyle helperStyle; public readonly TextStyle hintStyle; public readonly TextStyle errorStyle; public readonly int? errorMaxLines; public readonly bool? hasFloatingPlaceholder; public readonly bool? isDense; public readonly EdgeInsets contentPadding; public readonly bool? isCollapsed; public readonly TextStyle prefixStyle; public readonly TextStyle suffixStyle; public readonly TextStyle counterStyle; public readonly bool? filled; public readonly Color fillColor; public readonly InputBorder errorBorder; public readonly InputBorder focusedBorder; public readonly InputBorder focusedErrorBorder; public readonly InputBorder disabledBorder; public readonly InputBorder enabledBorder; public readonly InputBorder border; public readonly bool alignLabelWithHint; public override void debugFillProperties(DiagnosticPropertiesBuilder properties) { base.debugFillProperties(properties); InputDecorationTheme defaultTheme = new InputDecorationTheme(); properties.add(new DiagnosticsProperty("labelStyle", this.labelStyle, defaultValue: defaultTheme.labelStyle)); properties.add(new DiagnosticsProperty("helperStyle", this.helperStyle, defaultValue: defaultTheme.helperStyle)); properties.add(new DiagnosticsProperty("hintStyle", this.hintStyle, defaultValue: defaultTheme.hintStyle)); properties.add(new DiagnosticsProperty("errorStyle", this.errorStyle, defaultValue: defaultTheme.errorStyle)); properties.add(new DiagnosticsProperty("errorMaxLines", this.errorMaxLines, defaultValue: defaultTheme.errorMaxLines)); properties.add(new DiagnosticsProperty("hasFloatingPlaceholder", this.hasFloatingPlaceholder, defaultValue: defaultTheme.hasFloatingPlaceholder)); properties.add(new DiagnosticsProperty("isDense", this.isDense, defaultValue: defaultTheme.isDense)); properties.add(new DiagnosticsProperty("contentPadding", this.contentPadding, defaultValue: defaultTheme.contentPadding)); properties.add(new DiagnosticsProperty("isCollapsed", this.isCollapsed, defaultValue: defaultTheme.isCollapsed)); properties.add(new DiagnosticsProperty("prefixStyle", this.prefixStyle, defaultValue: defaultTheme.prefixStyle)); properties.add(new DiagnosticsProperty("suffixStyle", this.suffixStyle, defaultValue: defaultTheme.suffixStyle)); properties.add(new DiagnosticsProperty("counterStyle", this.counterStyle, defaultValue: defaultTheme.counterStyle)); properties.add(new DiagnosticsProperty("filled", this.filled, defaultValue: defaultTheme.filled)); properties.add(new DiagnosticsProperty("fillColor", this.fillColor, defaultValue: defaultTheme.fillColor)); properties.add(new DiagnosticsProperty("errorBorder", this.errorBorder, defaultValue: defaultTheme.errorBorder)); properties.add(new DiagnosticsProperty("focusedBorder", this.focusedBorder, defaultValue: defaultTheme.focusedErrorBorder)); properties.add(new DiagnosticsProperty("focusedErrorBorder", this.focusedErrorBorder, defaultValue: defaultTheme.focusedErrorBorder)); properties.add(new DiagnosticsProperty("disabledBorder", this.disabledBorder, defaultValue: defaultTheme.disabledBorder)); properties.add(new DiagnosticsProperty("enabledBorder", this.enabledBorder, defaultValue: defaultTheme.enabledBorder)); properties.add( new DiagnosticsProperty("border", this.border, defaultValue: defaultTheme.border)); properties.add(new DiagnosticsProperty("alignLabelWithHint", this.alignLabelWithHint, defaultValue: defaultTheme.alignLabelWithHint)); } } }