using System; using System.Collections.Generic; using System.Linq; using RSG; using UIWidgets.animation; using UIWidgets.foundation; using UIWidgets.ui; namespace UIWidgets.widgets { public class ScrollController : ChangeNotifier { public ScrollController( double initialScrollOffset = 0.0, bool keepScrollOffset = true, string debugLabel = null ) { this._initialScrollOffset = initialScrollOffset; this.keepScrollOffset = keepScrollOffset; this.debugLabel = debugLabel; } public virtual double initialScrollOffset { get { return this._initialScrollOffset; } } readonly double _initialScrollOffset; public readonly bool keepScrollOffset; public readonly string debugLabel; public IEnumerable positions { get { return this._positions; } } readonly List _positions = new List(); public bool hasClients { get { return this._positions.isNotEmpty(); } } public ScrollPosition position { get { D.assert(this._positions.isNotEmpty(), "ScrollController not attached to any scroll views."); D.assert(this._positions.Count == 1, "ScrollController attached to multiple scroll views."); return this._positions.Single(); } } public double offset { get { return this.position.pixels; } } public IPromise animateTo(double to, TimeSpan duration, Curve curve ) { D.assert(this._positions.isNotEmpty(), "ScrollController not attached to any scroll views."); List animations = Enumerable.Repeat((IPromise) null, this._positions.Count).ToList(); for (int i = 0; i < this._positions.Count; i += 1) { animations[i] = this._positions[i].animateTo(to, duration: duration, curve: curve); } return Promise.All(animations); } public void jumpTo(double value) { D.assert(this._positions.isNotEmpty(), "ScrollController not attached to any scroll views."); foreach (ScrollPosition position in new List(this._positions)) { position.jumpTo(value); } } public virtual void attach(ScrollPosition position) { D.assert(!this._positions.Contains(position)); this._positions.Add(position); position.addListener(this.notifyListeners); } public virtual void detach(ScrollPosition position) { D.assert(this._positions.Contains(position)); position.removeListener(this.notifyListeners); this._positions.Remove(position); } public override void dispose() { foreach (ScrollPosition position in this._positions) { position.removeListener(this.notifyListeners); } base.dispose(); } public ScrollPosition createScrollPosition( ScrollPhysics physics, ScrollContext context, ScrollPosition oldPosition ) { return new ScrollPositionWithSingleContext( physics: physics, context: context, initialPixels: this.initialScrollOffset, keepScrollOffset: this.keepScrollOffset, oldPosition: oldPosition, debugLabel: this.debugLabel ); } public override string ToString() { List description = new List(); this.debugFillDescription(description); return string.Format("{0}({1})", Diagnostics.describeIdentity(this), string.Join(", ", description.ToArray())); } protected virtual void debugFillDescription(List description) { if (this.debugLabel != null) { description.Add(this.debugLabel); } if (this.initialScrollOffset != 0.0) { description.Add(string.Format("initialScrollOffset: {0:F1}, ", this.initialScrollOffset)); } if (this._positions.isEmpty()) { description.Add("no clients"); } else if (this._positions.Count == 1) { description.Add(string.Format("one client, offset {0:F1}", this.offset)); } else { description.Add(this._positions.Count + " clients"); } } } public class TrackingScrollController : ScrollController { public TrackingScrollController( double initialScrollOffset = 0.0, bool keepScrollOffset = true, String debugLabel = null ) : base(initialScrollOffset: initialScrollOffset, keepScrollOffset: keepScrollOffset, debugLabel: debugLabel) { } readonly Dictionary _positionToListener = new Dictionary(); ScrollPosition _lastUpdated; double? _lastUpdatedOffset; public ScrollPosition mostRecentlyUpdatedPosition { get { return this._lastUpdated; } } public override double initialScrollOffset { get { return this._lastUpdatedOffset ?? base.initialScrollOffset; } } public override void attach(ScrollPosition position) { base.attach(position); D.assert(!this._positionToListener.ContainsKey(position)); this._positionToListener[position] = () => { this._lastUpdated = position; this._lastUpdatedOffset = position.pixels; }; position.addListener(this._positionToListener[position]); } public override void detach(ScrollPosition position) { base.detach(position); D.assert(this._positionToListener.ContainsKey(position)); position.removeListener(this._positionToListener[position]); this._positionToListener.Remove(position); if (this._lastUpdated == position) { this._lastUpdated = null; } if (this._positionToListener.isEmpty()) { this._lastUpdatedOffset = null; } } public override void dispose() { foreach (ScrollPosition position in this.positions) { D.assert(this._positionToListener.ContainsKey(position)); position.removeListener(this._positionToListener[position]); } base.dispose(); } } }