using System; using System.Collections.Generic; using RSG; using Unity.UIWidgets.animation; using Unity.UIWidgets.foundation; using Unity.UIWidgets.painting; using Unity.UIWidgets.physics; using Unity.UIWidgets.rendering; using Unity.UIWidgets.scheduler; using Unity.UIWidgets.ui; using UnityEngine; namespace Unity.UIWidgets.widgets { public interface ListWheelChildDelegate { Widget build(BuildContext context, int index); int? estimatedChildCount { get; } int trueIndexOf(int index); bool shouldRebuild(ListWheelChildDelegate oldDelegate); } public class ListWheelChildListDelegate : ListWheelChildDelegate { public ListWheelChildListDelegate( List children ) { D.assert(children != null); this.children = children; } public readonly List children; public int? estimatedChildCount { get { return this.children.Count; } } public Widget build(BuildContext context, int index) { if (index < 0 || index >= this.children.Count) { return null; } return new Container(child: this.children[index]); } public int trueIndexOf(int index) { return index; } public bool shouldRebuild(ListWheelChildDelegate oldDelegate) { return this.children != ((ListWheelChildListDelegate) oldDelegate).children; } } public class ListWheelChildLoopingListDelegate : ListWheelChildDelegate { public ListWheelChildLoopingListDelegate( List children ) { D.assert(children != null); this.children = children; } public readonly List children; public int? estimatedChildCount { get { return null; } } public int trueIndexOf(int index) { while (index < 0) { index += this.children.Count; } return index % this.children.Count; } public Widget build(BuildContext context, int index) { if (this.children.isEmpty()) { return null; } while (index < 0) { index += this.children.Count; } return new Container(child: this.children[index % this.children.Count]); } public bool shouldRebuild(ListWheelChildDelegate oldDelegate) { return this.children != ((ListWheelChildLoopingListDelegate) oldDelegate).children; } } public class ListWheelChildBuilderDelegate : ListWheelChildDelegate { public ListWheelChildBuilderDelegate( IndexedWidgetBuilder builder, int? childCount = null ) { D.assert(builder != null); this.builder = builder; this.childCount = childCount; } public readonly IndexedWidgetBuilder builder; public readonly int? childCount; public int? estimatedChildCount { get { return this.childCount; } } public Widget build(BuildContext context, int index) { if (this.childCount == null) { Widget child = this.builder(context, index); return child == null ? null : new Container(child: child); } if (index < 0 || index >= this.childCount) { return null; } return new Container(child: this.builder(context, index)); } public int trueIndexOf(int index) { return index; } public bool shouldRebuild(ListWheelChildDelegate oldDelegate) { return this.builder != ((ListWheelChildBuilderDelegate) oldDelegate).builder || this.childCount != ((ListWheelChildBuilderDelegate) oldDelegate).childCount; } } class ListWheelScrollViewUtils { public static int _getItemFromOffset( float offset, float itemExtent, float minScrollExtent, float maxScrollExtent ) { return (_clipOffsetToScrollableRange(offset, minScrollExtent, maxScrollExtent) / itemExtent).round(); } public static float _clipOffsetToScrollableRange( float offset, float minScrollExtent, float maxScrollExtent ) { return Mathf.Min(Mathf.Max(offset, minScrollExtent), maxScrollExtent); } } public class FixedExtentScrollController : ScrollController { public FixedExtentScrollController( int initialItem = 0 ) { this.initialItem = initialItem; } public readonly int initialItem; public int selectedItem { get { D.assert(this.positions.isNotEmpty(), () => "FixedExtentScrollController.selectedItem cannot be accessed before a scroll view is built with it." ); D.assert(this.positions.Count == 1, () => "The selectedItem property cannot be read when multiple scroll views are attached to the same FixedExtentScrollController." ); _FixedExtentScrollPosition position = (_FixedExtentScrollPosition) this.position; return position.itemIndex; } } public IPromise animateToItem( int itemIndex, TimeSpan duration, Curve curve ) { if (!this.hasClients) { return Promise.Resolved(); } List futures = new List(); foreach (_FixedExtentScrollPosition position in this.positions) { futures.Add(position.animateTo( itemIndex * position.itemExtent, duration: duration, curve: curve )); } return Promise.All(futures); } public void jumpToItem(int itemIndex) { foreach (_FixedExtentScrollPosition position in this.positions) { position.jumpTo(itemIndex * position.itemExtent); } } public override ScrollPosition createScrollPosition(ScrollPhysics physics, ScrollContext context, ScrollPosition oldPosition) { return new _FixedExtentScrollPosition( physics: physics, context: context, initialItem: this.initialItem, oldPosition: oldPosition ); } } public interface IFixedExtentMetrics { int itemIndex { set; get; } FixedExtentMetrics copyWith( float? minScrollExtent = null, float? maxScrollExtent = null, float? pixels = null, float? viewportDimension = null, AxisDirection? axisDirection = null, int? itemIndex = null ); } public class FixedExtentMetrics : FixedScrollMetrics, IFixedExtentMetrics { public FixedExtentMetrics( int itemIndex, float minScrollExtent = 0.0f, float maxScrollExtent = 0.0f, float pixels = 0.0f, float viewportDimension = 0.0f, AxisDirection axisDirection = AxisDirection.down ) : base( minScrollExtent: minScrollExtent, maxScrollExtent: maxScrollExtent, pixels: pixels, viewportDimension: viewportDimension, axisDirection: axisDirection ) { this.itemIndex = itemIndex; } public int itemIndex { get; set; } public FixedExtentMetrics copyWith( float? minScrollExtent = null, float? maxScrollExtent = null, float? pixels = null, float? viewportDimension = null, AxisDirection? axisDirection = null, int? itemIndex = null ) { return new FixedExtentMetrics( minScrollExtent: minScrollExtent ?? this.minScrollExtent, maxScrollExtent: maxScrollExtent ?? this.maxScrollExtent, pixels: pixels ?? this.pixels, viewportDimension: viewportDimension ?? this.viewportDimension, axisDirection: axisDirection ?? this.axisDirection, itemIndex: itemIndex ?? this.itemIndex ); } } class _FixedExtentScrollPosition : ScrollPositionWithSingleContext, IFixedExtentMetrics { public _FixedExtentScrollPosition( ScrollPhysics physics, ScrollContext context, int initialItem, bool keepScrollOffset = true, ScrollPosition oldPosition = null, string debugLabel = null ) : base( physics: physics, context: context, initialPixels: _getItemExtentFromScrollContext(context) * initialItem, keepScrollOffset: keepScrollOffset, oldPosition: oldPosition, debugLabel: debugLabel ) { D.assert( context is _FixedExtentScrollableState, () => "FixedExtentScrollController can only be used with ListWheelScrollViews" ); } static float _getItemExtentFromScrollContext(ScrollContext context) { _FixedExtentScrollableState scrollable = (_FixedExtentScrollableState) context; return scrollable.itemExtent; } public float itemExtent { get { return _getItemExtentFromScrollContext(this.context); } } public int itemIndex { get { return ListWheelScrollViewUtils._getItemFromOffset( offset: this.pixels, itemExtent: this.itemExtent, minScrollExtent: this.minScrollExtent, maxScrollExtent: this.maxScrollExtent ); } set { } } public FixedExtentMetrics copyWith( float? minScrollExtent = null, float? maxScrollExtent = null, float? pixels = null, float? viewportDimension = null, AxisDirection? axisDirection = null, int? itemIndex = null ) { return new FixedExtentMetrics( minScrollExtent: minScrollExtent ?? this.minScrollExtent, maxScrollExtent: maxScrollExtent ?? this.maxScrollExtent, pixels: pixels ?? this.pixels, viewportDimension: viewportDimension ?? this.viewportDimension, axisDirection: axisDirection ?? this.axisDirection, itemIndex: itemIndex ?? this.itemIndex ); } } class _FixedExtentScrollable : Scrollable { public _FixedExtentScrollable( float itemExtent, ViewportBuilder viewportBuilder, Key key = null, AxisDirection axisDirection = AxisDirection.down, ScrollController controller = null, ScrollPhysics physics = null ) : base( key: key, axisDirection: axisDirection, controller: controller, physics: physics, viewportBuilder: viewportBuilder ) { this.itemExtent = itemExtent; } public readonly float itemExtent; public override State createState() { return new _FixedExtentScrollableState(); } } class _FixedExtentScrollableState : ScrollableState { public float itemExtent { get { _FixedExtentScrollable actualWidget = (_FixedExtentScrollable) this.widget; return actualWidget.itemExtent; } } } public class FixedExtentScrollPhysics : ScrollPhysics { public FixedExtentScrollPhysics( ScrollPhysics parent = null ) : base(parent: parent) { } public override ScrollPhysics applyTo(ScrollPhysics ancestor) { return new FixedExtentScrollPhysics(parent: this.buildParent(ancestor)); } public override Simulation createBallisticSimulation(ScrollMetrics position, float velocity) { D.assert( position is _FixedExtentScrollPosition, () => "FixedExtentScrollPhysics can only be used with Scrollables that uses " + "the FixedExtentScrollController" ); _FixedExtentScrollPosition metrics = (_FixedExtentScrollPosition) position; if ((velocity <= 0.0f && metrics.pixels <= metrics.minScrollExtent) || (velocity >= 0.0f && metrics.pixels >= metrics.maxScrollExtent)) { return base.createBallisticSimulation(metrics, velocity); } Simulation testFrictionSimulation = base.createBallisticSimulation(metrics, velocity); if (testFrictionSimulation != null && (testFrictionSimulation.x(float.PositiveInfinity) == metrics.minScrollExtent || testFrictionSimulation.x(float.PositiveInfinity) == metrics.maxScrollExtent)) { return base.createBallisticSimulation(metrics, velocity); } int settlingItemIndex = ListWheelScrollViewUtils._getItemFromOffset( offset: testFrictionSimulation?.x(float.PositiveInfinity) ?? metrics.pixels, itemExtent: metrics.itemExtent, minScrollExtent: metrics.minScrollExtent, maxScrollExtent: metrics.maxScrollExtent ); float settlingPixels = settlingItemIndex * metrics.itemExtent; if (velocity.abs() < this.tolerance.velocity && (settlingPixels - metrics.pixels).abs() < this.tolerance.distance) { return null; } if (settlingItemIndex == metrics.itemIndex) { return new SpringSimulation(this.spring, metrics.pixels, settlingPixels, velocity, tolerance: this.tolerance ); } return FrictionSimulation.through( metrics.pixels, settlingPixels, velocity, this.tolerance.velocity * velocity.sign() ); } } public class ListWheelScrollView : StatefulWidget { public ListWheelScrollView( float itemExtent, List children = null, Key key = null, ScrollController controller = null, ScrollPhysics physics = null, float diameterRatio = RenderListWheelViewport.defaultDiameterRatio, float perspective = RenderListWheelViewport.defaultPerspective, float offAxisFraction = 0.0f, bool useMagnifier = false, float magnification = 1.0f, ValueChanged onSelectedItemChanged = null, bool clipToSize = true, bool renderChildrenOutsideViewport = false, ListWheelChildDelegate childDelegate = null ) : base(key: key) { D.assert(children != null || childDelegate != null); D.assert(diameterRatio > 0.0, () => RenderListWheelViewport.diameterRatioZeroMessage); D.assert(perspective > 0); D.assert(perspective <= 0.01f, () => RenderListWheelViewport.perspectiveTooHighMessage); D.assert(magnification > 0); D.assert(itemExtent > 0); D.assert( !renderChildrenOutsideViewport || !clipToSize, () => RenderListWheelViewport.clipToSizeAndRenderChildrenOutsideViewportConflict ); this.childDelegate = childDelegate ?? new ListWheelChildListDelegate(children: children); this.itemExtent = itemExtent; this.controller = controller; this.physics = physics; this.diameterRatio = diameterRatio; this.perspective = perspective; this.offAxisFraction = offAxisFraction; this.useMagnifier = useMagnifier; this.magnification = magnification; this.onSelectedItemChanged = onSelectedItemChanged; this.clipToSize = clipToSize; this.renderChildrenOutsideViewport = renderChildrenOutsideViewport; } public static ListWheelScrollView useDelegate( float itemExtent, List children = null, ListWheelChildDelegate childDelegate = null, Key key = null, ScrollController controller = null, ScrollPhysics physics = null, float diameterRatio = RenderListWheelViewport.defaultDiameterRatio, float perspective = RenderListWheelViewport.defaultPerspective, float offAxisFraction = 0.0f, bool useMagnifier = false, float magnification = 1.0f, ValueChanged onSelectedItemChanged = null, bool clipToSize = true, bool renderChildrenOutsideViewport = false ) { return new ListWheelScrollView( itemExtent: itemExtent, children: children, childDelegate: childDelegate, key: key, controller: controller, physics: physics, diameterRatio: diameterRatio, perspective: perspective, offAxisFraction: offAxisFraction, useMagnifier: useMagnifier, magnification: magnification, onSelectedItemChanged: onSelectedItemChanged, clipToSize: clipToSize, renderChildrenOutsideViewport: renderChildrenOutsideViewport ); } public readonly ScrollController controller; public readonly ScrollPhysics physics; public readonly float diameterRatio; public readonly float perspective; public readonly float offAxisFraction; public readonly bool useMagnifier; public readonly float magnification; public readonly float itemExtent; public readonly ValueChanged onSelectedItemChanged; public readonly bool clipToSize; public readonly bool renderChildrenOutsideViewport; public readonly ListWheelChildDelegate childDelegate; public override State createState() { return new _ListWheelScrollViewState(); } } class _ListWheelScrollViewState : State { int _lastReportedItemIndex = 0; ScrollController scrollController; public override void initState() { base.initState(); this.scrollController = this.widget.controller ?? new FixedExtentScrollController(); if (this.widget.controller is FixedExtentScrollController controller) { this._lastReportedItemIndex = controller.initialItem; } } public override void didUpdateWidget(StatefulWidget oldWidget) { base.didUpdateWidget(oldWidget); if (this.widget.controller != null && this.widget.controller != this.scrollController) { ScrollController oldScrollController = this.scrollController; SchedulerBinding.instance.addPostFrameCallback((_) => { oldScrollController.dispose(); }); this.scrollController = this.widget.controller; } } public override Widget build(BuildContext context) { return new NotificationListener( onNotification: (ScrollNotification notification) => { if (notification.depth == 0 && this.widget.onSelectedItemChanged != null && notification is ScrollUpdateNotification && notification.metrics is FixedExtentMetrics metrics) { int currentItemIndex = metrics.itemIndex; if (currentItemIndex != this._lastReportedItemIndex) { this._lastReportedItemIndex = currentItemIndex; int trueIndex = this.widget.childDelegate.trueIndexOf(currentItemIndex); this.widget.onSelectedItemChanged(trueIndex); } } return false; }, child: new _FixedExtentScrollable( controller: this.scrollController, physics: this.widget.physics, itemExtent: this.widget.itemExtent, viewportBuilder: (BuildContext _context, ViewportOffset _offset) => { return new ListWheelViewport( diameterRatio: this.widget.diameterRatio, perspective: this.widget.perspective, offAxisFraction: this.widget.offAxisFraction, useMagnifier: this.widget.useMagnifier, magnification: this.widget.magnification, itemExtent: this.widget.itemExtent, clipToSize: this.widget.clipToSize, renderChildrenOutsideViewport: this.widget.renderChildrenOutsideViewport, offset: _offset, childDelegate: this.widget.childDelegate ); } ) ); } } public class ListWheelElement : RenderObjectElement, IListWheelChildManager { public ListWheelElement(ListWheelViewport widget) : base(widget) { } public new ListWheelViewport widget { get { return (ListWheelViewport) base.widget; } } public new RenderListWheelViewport renderObject { get { return (RenderListWheelViewport) base.renderObject; } } readonly Dictionary _childWidgets = new Dictionary(); readonly SplayTree _childElements = new SplayTree(); public override void update(Widget newWidget) { ListWheelViewport oldWidget = this.widget; base.update(newWidget); ListWheelChildDelegate newDelegate = ((ListWheelViewport) newWidget).childDelegate; ListWheelChildDelegate oldDelegate = oldWidget.childDelegate; if (newDelegate != oldDelegate && (newDelegate.GetType() != oldDelegate.GetType() || newDelegate.shouldRebuild(oldDelegate))) { this.performRebuild(); } } public int? childCount { get { return this.widget.childDelegate.estimatedChildCount; } } protected override void performRebuild() { this._childWidgets.Clear(); base.performRebuild(); if (this._childElements.isEmpty()) { return; } int firstIndex = this._childElements.First().Key; int lastIndex = this._childElements.Last().Key; for (int index = firstIndex; index <= lastIndex; ++index) { Element newChild = this.updateChild(this._childElements[index], this.retrieveWidget(index), index); if (newChild != null) { this._childElements[index] = newChild; } else { this._childElements.Remove(index); } } } Widget retrieveWidget(int index) { return this._childWidgets.putIfAbsent(index, () => { return this.widget.childDelegate.build(this, index); }); } public bool childExistsAt(int index) { return this.retrieveWidget(index) != null; } public void createChild(int index, RenderBox after) { this.owner.buildScope(this, () => { bool insertFirst = after == null; D.assert(insertFirst || this._childElements[index - 1] != null); // Debug.Log($"{index}: {this._childElements.getOrDefault(index)}"); Element newChild = this.updateChild(this._childElements.getOrDefault(index), this.retrieveWidget(index), index); // Debug.Log(newChild); if (newChild != null) { this._childElements[index] = newChild; } else { this._childElements.Remove(index); } }); } public void removeChild(RenderBox child) { int index = this.renderObject.indexOf(child); this.owner.buildScope(this, () => { D.assert(this._childElements.ContainsKey(index)); Element result = this.updateChild(this._childElements[index], null, index); D.assert(result == null); this._childElements.Remove(index); D.assert(!this._childElements.ContainsKey(index)); }); } protected override Element updateChild(Element child, Widget newWidget, object newSlot) { ListWheelParentData oldParentData = (ListWheelParentData) child?.renderObject?.parentData; Element newChild = base.updateChild(child, newWidget, newSlot); ListWheelParentData newParentData = (ListWheelParentData) newChild?.renderObject?.parentData; if (newParentData != null) { newParentData.index = (int) newSlot; if (oldParentData != null) { newParentData.offset = oldParentData.offset; } } return newChild; } protected override void insertChildRenderObject(RenderObject child, object slot) { RenderListWheelViewport renderObject = this.renderObject; D.assert(renderObject.debugValidateChild(child)); renderObject.insert((RenderBox) child, (RenderBox) this._childElements.getOrDefault((int) slot - 1)?.renderObject); // Debug.Log($"insert: {this._childElements.getOrDefault((int) slot - 1)}"); D.assert(renderObject == this.renderObject); } protected override void moveChildRenderObject(RenderObject child, dynamic slot) { const string moveChildRenderObjectErrorMessage = "Currently we maintain the list in contiguous increasing order, so " + "moving children around is not allowed."; D.assert(false, () => moveChildRenderObjectErrorMessage); } protected override void removeChildRenderObject(RenderObject child) { D.assert(child.parent == this.renderObject); this.renderObject.remove((RenderBox) child); } public override void visitChildren(ElementVisitor visitor) { foreach (var item in this._childElements) { visitor(item.Value); } } protected override void forgetChild(Element child) { this._childElements.Remove((int) (child.slot)); } } public class ListWheelViewport : RenderObjectWidget { public ListWheelViewport( float itemExtent, ViewportOffset offset, ListWheelChildDelegate childDelegate, Key key = null, float diameterRatio = RenderListWheelViewport.defaultDiameterRatio, float perspective = RenderListWheelViewport.defaultPerspective, float offAxisFraction = 0.0f, bool useMagnifier = false, float magnification = 1.0f, bool clipToSize = true, bool renderChildrenOutsideViewport = false ) : base(key: key) { D.assert(childDelegate != null); D.assert(offset != null); D.assert(diameterRatio > 0, () => RenderListWheelViewport.diameterRatioZeroMessage); D.assert(perspective > 0); D.assert(perspective <= 0.01, () => RenderListWheelViewport.perspectiveTooHighMessage); D.assert(itemExtent > 0); D.assert( !renderChildrenOutsideViewport || !clipToSize, () => RenderListWheelViewport.clipToSizeAndRenderChildrenOutsideViewportConflict ); this.itemExtent = itemExtent; this.offset = offset; this.childDelegate = childDelegate; this.diameterRatio = diameterRatio; this.perspective = perspective; this.offAxisFraction = offAxisFraction; this.useMagnifier = useMagnifier; this.magnification = magnification; this.clipToSize = clipToSize; this.renderChildrenOutsideViewport = renderChildrenOutsideViewport; } public readonly float diameterRatio; public readonly float perspective; public readonly float offAxisFraction; public readonly bool useMagnifier; public readonly float magnification; public readonly float itemExtent; public readonly bool clipToSize; public readonly bool renderChildrenOutsideViewport; public readonly ViewportOffset offset; public readonly ListWheelChildDelegate childDelegate; public override Element createElement() { return new ListWheelElement(this); } public override RenderObject createRenderObject(BuildContext context) { ListWheelElement childManager = (ListWheelElement) context; return new RenderListWheelViewport( childManager: childManager, offset: this.offset, diameterRatio: this.diameterRatio, perspective: this.perspective, offAxisFraction: this.offAxisFraction, useMagnifier: this.useMagnifier, magnification: this.magnification, itemExtent: this.itemExtent, clipToSize: this.clipToSize, renderChildrenOutsideViewport: this.renderChildrenOutsideViewport ); } public override void updateRenderObject(BuildContext context, RenderObject renderObject) { var viewport = (RenderListWheelViewport) renderObject; viewport.offset = this.offset; viewport.diameterRatio = this.diameterRatio; viewport.perspective = this.perspective; viewport.offAxisFraction = this.offAxisFraction; viewport.useMagnifier = this.useMagnifier; viewport.magnification = this.magnification; viewport.itemExtent = this.itemExtent; viewport.clipToSize = this.clipToSize; viewport.renderChildrenOutsideViewport = this.renderChildrenOutsideViewport; } } }