using System; using System.Collections.Generic; using Unity.UIWidgets.animation; using Unity.UIWidgets.async2; 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; using Canvas = Unity.UIWidgets.ui.Canvas; using Color = Unity.UIWidgets.ui.Color; using Rect = Unity.UIWidgets.ui.Rect; namespace Unity.UIWidgets.widgets { public class GlowingOverscrollIndicator : StatefulWidget { public GlowingOverscrollIndicator( Key key = null, bool showLeading = true, bool showTrailing = true, AxisDirection axisDirection = AxisDirection.up, Color color = null, ScrollNotificationPredicate notificationPredicate = null, Widget child = null ) : base(key: key) { D.assert(color != null); this.showLeading = showLeading; this.showTrailing = showTrailing; this.axisDirection = axisDirection; this.child = child; this.color = color; this.notificationPredicate = notificationPredicate ?? ScrollNotification.defaultScrollNotificationPredicate; } public readonly bool showLeading; public readonly bool showTrailing; public readonly AxisDirection axisDirection; public Axis axis { get { return AxisUtils.axisDirectionToAxis(axisDirection); } } public readonly Color color; public readonly ScrollNotificationPredicate notificationPredicate; public readonly Widget child; public override State createState() { return new _GlowingOverscrollIndicatorState(); } public override void debugFillProperties(DiagnosticPropertiesBuilder properties) { base.debugFillProperties(properties); properties.add(new EnumProperty("axisDirection", axisDirection)); string showDescription; if (showLeading && showTrailing) { showDescription = "both sides"; } else if (showLeading) { showDescription = "leading side only"; } else if (showTrailing) { showDescription = "trailing side only"; } else { showDescription = "neither side (!)"; } properties.add(new MessageProperty("show", showDescription)); properties.add(new DiagnosticsProperty("color", color, showName: false)); } } class _GlowingOverscrollIndicatorState : TickerProviderStateMixin { _GlowController _leadingController; _GlowController _trailingController; Listenable _leadingAndTrailingListener; public override void initState() { base.initState(); _leadingController = new _GlowController(vsync: this, color: widget.color, axis: widget.axis); _trailingController = new _GlowController(vsync: this, color: widget.color, axis: widget.axis); _leadingAndTrailingListener = ListenableUtils.merge(new List {_leadingController, _trailingController}); } public override void didUpdateWidget(StatefulWidget _oldWidget) { base.didUpdateWidget(_oldWidget); GlowingOverscrollIndicator oldWidget = _oldWidget as GlowingOverscrollIndicator; if (oldWidget.color != widget.color || oldWidget.axis != widget.axis) { _leadingController.color = widget.color; _leadingController.axis = widget.axis; _trailingController.color = widget.color; _trailingController.axis = widget.axis; } } Type _lastNotificationType; Dictionary _accepted = new Dictionary {{false, true}, {true, true}}; bool _handleScrollNotification(ScrollNotification notification) { if (!widget.notificationPredicate(notification)) { return false; } if (notification is OverscrollNotification) { _GlowController controller; OverscrollNotification _notification = notification as OverscrollNotification; if (_notification.overscroll < 0.0f) { controller = _leadingController; } else if (_notification.overscroll > 0.0f) { controller = _trailingController; } else { throw new Exception("overscroll is 0.0f!"); } bool isLeading = controller == _leadingController; if (_lastNotificationType != typeof(OverscrollNotification)) { OverscrollIndicatorNotification confirmationNotification = new OverscrollIndicatorNotification(leading: isLeading); confirmationNotification.dispatch(context); _accepted[isLeading] = confirmationNotification._accepted; } D.assert(controller != null); D.assert(_notification.metrics.axis() == widget.axis); if (_accepted[isLeading]) { if (_notification.velocity != 0.0f) { D.assert(_notification.dragDetails == null); controller.absorbImpact(_notification.velocity.abs()); } else { D.assert(_notification.overscroll != 0.0f); if (_notification.dragDetails != null) { D.assert(_notification.dragDetails.globalPosition != null); RenderBox renderer = (RenderBox) _notification.context.findRenderObject(); D.assert(renderer != null); D.assert(renderer.hasSize); Size size = renderer.size; Offset position = renderer.globalToLocal(_notification.dragDetails.globalPosition); switch (_notification.metrics.axis()) { case Axis.horizontal: controller.pull(_notification.overscroll.abs(), size.width, position.dy.clamp(0.0f, size.height), size.height); break; case Axis.vertical: controller.pull(_notification.overscroll.abs(), size.height, position.dx.clamp(0.0f, size.width), size.width); break; } } } } } else if (notification is ScrollEndNotification || notification is ScrollUpdateNotification) { if ((notification as ScrollEndNotification).dragDetails != null) { _leadingController.scrollEnd(); _trailingController.scrollEnd(); } } _lastNotificationType = notification.GetType(); return false; } public override void dispose() { _leadingController.dispose(); _trailingController.dispose(); base.dispose(); } public override Widget build(BuildContext context) { return new NotificationListener( onNotification: _handleScrollNotification, child: new RepaintBoundary( child: new CustomPaint( foregroundPainter: new _GlowingOverscrollIndicatorPainter( leadingController: widget.showLeading ? _leadingController : null, trailingController: widget.showTrailing ? _trailingController : null, axisDirection: widget.axisDirection, repaint: _leadingAndTrailingListener ), child: new RepaintBoundary( child: widget.child ) ) ) ); } } enum _GlowState { idle, absorb, pull, recede } class _GlowController : ChangeNotifier { public _GlowController( TickerProvider vsync, Color color, Axis axis ) { D.assert(vsync != null); D.assert(color != null); _color = color; _axis = axis; _glowController = new AnimationController(vsync: vsync); _glowController.addStatusListener(_changePhase); Animation decelerator = new CurvedAnimation( parent: _glowController, curve: Curves.decelerate ); decelerator.addListener(notifyListeners); _glowOpacity = decelerator.drive(_glowOpacityTween); _glowSize = decelerator.drive(_glowSizeTween); _displacementTicker = vsync.createTicker(_tickDisplacement); } _GlowState _state = _GlowState.idle; AnimationController _glowController; Timer _pullRecedeTimer; FloatTween _glowOpacityTween = new FloatTween(begin: 0.0f, end: 0.0f); Animation _glowOpacity; FloatTween _glowSizeTween = new FloatTween(begin: 0.0f, end: 0.0f); Animation _glowSize; Ticker _displacementTicker; TimeSpan? _displacementTickerLastElapsed; float _displacementTarget = 0.5f; float _displacement = 0.5f; float _pullDistance = 0.0f; public Color color { get { return _color; } set { D.assert(color != null); if (color == value) { return; } _color = value; notifyListeners(); } } Color _color; public Axis axis { get { return _axis; } set { if (axis == value) { return; } _axis = value; notifyListeners(); } } Axis _axis; readonly TimeSpan _recedeTime = new TimeSpan(0, 0, 0, 0, 600); readonly TimeSpan _pullTime = new TimeSpan(0, 0, 0, 0, 167); readonly TimeSpan _pullHoldTime = new TimeSpan(0, 0, 0, 0, 167); readonly TimeSpan _pullDecayTime = new TimeSpan(0, 0, 0, 0, 2000); static readonly TimeSpan _crossAxisHalfTime = new TimeSpan(0, 0, 0, 0, (1000.0f / 60.0f).round()); const float _maxOpacity = 0.5f; const float _pullOpacityGlowFactor = 0.8f; const float _velocityGlowFactor = 0.00006f; const float _sqrt3 = 1.73205080757f; // Mathf.Sqrt(3) const float _widthToHeightFactor = (3.0f / 4.0f) * (2.0f - _sqrt3); const float _minVelocity = 100.0f; // logical pixels per second const float _maxVelocity = 10000.0f; // logical pixels per second public override void dispose() { _glowController.dispose(); _displacementTicker.dispose(); _pullRecedeTimer?.cancel(); base.dispose(); } public void absorbImpact(float velocity) { D.assert(velocity >= 0.0f); _pullRecedeTimer?.cancel(); _pullRecedeTimer = null; velocity = velocity.clamp(_minVelocity, _maxVelocity); _glowOpacityTween.begin = _state == _GlowState.idle ? 0.3f : _glowOpacity.value; _glowOpacityTween.end = (velocity * _velocityGlowFactor).clamp(_glowOpacityTween.begin, _maxOpacity); _glowSizeTween.begin = _glowSize.value; _glowSizeTween.end = Mathf.Min(0.025f + 7.5e-7f * velocity * velocity, 1.0f); _glowController.duration = new TimeSpan(0, 0, 0, 0, (0.15f + velocity * 0.02f).round()); _glowController.forward(from: 0.0f); _displacement = 0.5f; _state = _GlowState.absorb; } public void pull(float overscroll, float extent, float crossAxisOffset, float crossExtent) { _pullRecedeTimer?.cancel(); _pullDistance += overscroll / 200.0f; // This factor is magic. Not clear why we need it to match Android. _glowOpacityTween.begin = _glowOpacity.value; _glowOpacityTween.end = Mathf.Min(_glowOpacity.value + overscroll / extent * _pullOpacityGlowFactor, _maxOpacity); float height = Mathf.Min(extent, crossExtent * _widthToHeightFactor); _glowSizeTween.begin = _glowSize.value; _glowSizeTween.end = Mathf.Max(1.0f - 1.0f / (0.7f * Mathf.Sqrt(_pullDistance * height)), _glowSize.value); _displacementTarget = crossAxisOffset / crossExtent; if (_displacementTarget != _displacement) { if (!_displacementTicker.isTicking) { D.assert(_displacementTickerLastElapsed == null); _displacementTicker.start(); } } else { _displacementTicker.stop(); _displacementTickerLastElapsed = null; } _glowController.duration = _pullTime; if (_state != _GlowState.pull) { _glowController.forward(from: 0.0f); _state = _GlowState.pull; } else { if (!_glowController.isAnimating) { D.assert(_glowController.value == 1.0f); notifyListeners(); } } _pullRecedeTimer = Timer.create(_pullHoldTime, () => { _recede(_pullDecayTime); return null; }); } public void scrollEnd() { if (_state == _GlowState.pull) { _recede(_recedeTime); } } void _changePhase(AnimationStatus status) { if (status != AnimationStatus.completed) { return; } switch (_state) { case _GlowState.absorb: _recede(_recedeTime); break; case _GlowState.recede: _state = _GlowState.idle; _pullDistance = 0.0f; break; case _GlowState.pull: case _GlowState.idle: break; } } void _recede(TimeSpan duration) { if (_state == _GlowState.recede || _state == _GlowState.idle) { return; } _pullRecedeTimer?.cancel(); _pullRecedeTimer = null; _glowOpacityTween.begin = _glowOpacity.value; _glowOpacityTween.end = 0.0f; _glowSizeTween.begin = _glowSize.value; _glowSizeTween.end = 0.0f; _glowController.duration = duration; _glowController.forward(from: 0.0f); _state = _GlowState.recede; } void _tickDisplacement(TimeSpan elapsed) { if (_displacementTickerLastElapsed != null) { float? t = elapsed.Milliseconds - _displacementTickerLastElapsed?.Milliseconds; _displacement = _displacementTarget - (_displacementTarget - _displacement) * Mathf.Pow(2.0f, (-t ?? 0.0f) / _crossAxisHalfTime.Milliseconds); notifyListeners(); } if (PhysicsUtils.nearEqual(_displacementTarget, _displacement, Tolerance.defaultTolerance.distance)) { _displacementTicker.stop(); _displacementTickerLastElapsed = null; } else { _displacementTickerLastElapsed = elapsed; } } public void paint(Canvas canvas, Size size) { if (_glowOpacity.value == 0.0f) { return; } float baseGlowScale = size.width > size.height ? size.height / size.width : 1.0f; float radius = size.width * 3.0f / 2.0f; float height = Mathf.Min(size.height, size.width * _widthToHeightFactor); float scaleY = _glowSize.value * baseGlowScale; Rect rect = Rect.fromLTWH(0.0f, 0.0f, size.width, height); Offset center = new Offset((size.width / 2.0f) * (0.5f + _displacement), height - radius); Paint paint = new Paint(); paint.color = color.withOpacity(_glowOpacity.value); canvas.save(); canvas.scale(1.0f, scaleY); canvas.clipRect(rect); canvas.drawCircle(center, radius, paint); canvas.restore(); } } class _GlowingOverscrollIndicatorPainter : AbstractCustomPainter { public _GlowingOverscrollIndicatorPainter( _GlowController leadingController, _GlowController trailingController, AxisDirection axisDirection, Listenable repaint ) : base( repaint: repaint ) { this.leadingController = leadingController; this.trailingController = trailingController; this.axisDirection = axisDirection; } public readonly _GlowController leadingController; public readonly _GlowController trailingController; public readonly AxisDirection axisDirection; const float piOver2 = Mathf.PI / 2.0f; void _paintSide(Canvas canvas, Size size, _GlowController controller, AxisDirection axisDirection, GrowthDirection growthDirection) { if (controller == null) { return; } switch (GrowthDirectionUtils.applyGrowthDirectionToAxisDirection(axisDirection, growthDirection)) { case AxisDirection.up: controller.paint(canvas, size); break; case AxisDirection.down: canvas.save(); canvas.translate(0.0f, size.height); canvas.scale(1.0f, -1.0f); controller.paint(canvas, size); canvas.restore(); break; case AxisDirection.left: canvas.save(); canvas.rotate(piOver2); canvas.scale(1.0f, -1.0f); controller.paint(canvas, new Size(size.height, size.width)); canvas.restore(); break; case AxisDirection.right: canvas.save(); canvas.translate(size.width, 0.0f); canvas.rotate(piOver2); controller.paint(canvas, new Size(size.height, size.width)); canvas.restore(); break; } } public override void paint(Canvas canvas, Size size) { _paintSide(canvas, size, leadingController, axisDirection, GrowthDirection.reverse); _paintSide(canvas, size, trailingController, axisDirection, GrowthDirection.forward); } public override bool shouldRepaint(CustomPainter _oldDelegate) { _GlowingOverscrollIndicatorPainter oldDelegate = _oldDelegate as _GlowingOverscrollIndicatorPainter; return oldDelegate.leadingController != leadingController || oldDelegate.trailingController != trailingController; } } public class OverscrollIndicatorNotification : ViewportNotificationMixinNotification { public OverscrollIndicatorNotification( bool leading ) { this.leading = leading; } public readonly bool leading; internal bool _accepted = true; public void disallowGlow() { _accepted = false; } protected override void debugFillDescription(List description) { base.debugFillDescription(description); description.Add($"side: {(leading ? "leading edge" : "trailing edge")}"); } } }