using System; using System.Collections.Generic; using Unity.UIWidgets.foundation; using Unity.UIWidgets.ui; using UnityEngine; namespace Unity.UIWidgets.gestures { enum _DragState { ready, possible, accepted, } public delegate void GestureDragEndCallback(DragEndDetails details); public delegate void GestureDragCancelCallback(); public abstract class DragGestureRecognizer : OneSequenceGestureRecognizer { public DragGestureRecognizer( object debugOwner = null, PointerDeviceKind? kind = null, DragStartBehavior dragStartBehavior = DragStartBehavior.down) : base(debugOwner: debugOwner, kind: kind) { this.dragStartBehavior = dragStartBehavior; } public DragStartBehavior dragStartBehavior; public GestureDragDownCallback onDown; public GestureDragStartCallback onStart; public GestureDragUpdateCallback onUpdate; public GestureDragEndCallback onEnd; public GestureDragCancelCallback onCancel; public float? minFlingDistance; public float? minFlingVelocity; public float? maxFlingVelocity; _DragState _state = _DragState.ready; Offset _initialPosition; protected Offset _pendingDragOffset; TimeSpan _lastPendingEventTimestamp; protected abstract bool _isFlingGesture(VelocityEstimate estimate); protected abstract Offset _getDeltaForDetails(Offset delta); protected abstract float? _getPrimaryValueFromOffset(Offset value); protected abstract bool _hasSufficientPendingDragDeltaToAccept { get; } readonly Dictionary _velocityTrackers = new Dictionary(); public override void addScrollPointer(PointerScrollEvent evt) { this.startTrackingScrollerPointer(evt.pointer); if (this._state == _DragState.ready) { this._state = _DragState.possible; this._initialPosition = evt.position; if (this.onStart != null) { this.invokeCallback("onStart", () => { this.onStart(new DragStartDetails( sourceTimeStamp: evt.timeStamp, globalPosition: this._initialPosition )); return null; }); } } } public override void addAllowedPointer(PointerDownEvent evt) { this.startTrackingPointer(evt.pointer); this._velocityTrackers[evt.pointer] = new VelocityTracker(); if (this._state == _DragState.ready) { this._state = _DragState.possible; this._initialPosition = evt.position; this._pendingDragOffset = Offset.zero; this._lastPendingEventTimestamp = evt.timeStamp; if (this.onDown != null) { this.invokeCallback("onDown", () => { this.onDown(new DragDownDetails(globalPosition: this._initialPosition)); return null; }); } } else if (this._state == _DragState.accepted) { this.resolve(GestureDisposition.accepted); } } protected override void handleEvent(PointerEvent evt) { D.assert(this._state != _DragState.ready); if (evt is PointerScrollEvent) { Offset delta = evt.delta; if (this.onUpdate != null) { this.invokeCallback("onUpdate", () => { this.onUpdate(new DragUpdateDetails( sourceTimeStamp: evt.timeStamp, delta: this._getDeltaForDetails(delta), primaryDelta: this._getPrimaryValueFromOffset(delta), globalPosition: evt.position, isScroll: true )); return null; }); } this.stopTrackingScrollerPointer(evt.pointer); return; } if (!evt.synthesized && (evt is PointerDownEvent || evt is PointerMoveEvent)) { var tracker = this._velocityTrackers[evt.pointer]; D.assert(tracker != null); tracker.addPosition(evt.timeStamp, evt.position); } if (evt is PointerMoveEvent) { Offset delta = evt.delta; if (this._state == _DragState.accepted) { if (this.onUpdate != null) { this.invokeCallback("onUpdate", () => { this.onUpdate(new DragUpdateDetails( sourceTimeStamp: evt.timeStamp, delta: this._getDeltaForDetails(delta), primaryDelta: this._getPrimaryValueFromOffset(delta), globalPosition: evt.position )); return null; }); } } else { this._pendingDragOffset += delta; this._lastPendingEventTimestamp = evt.timeStamp; if (this._hasSufficientPendingDragDeltaToAccept) { this.resolve(GestureDisposition.accepted); } } } this.stopTrackingIfPointerNoLongerDown(evt); } public override void acceptGesture(int pointer) { if (this._state != _DragState.accepted) { this._state = _DragState.accepted; Offset delta = this._pendingDragOffset; var timestamp = this._lastPendingEventTimestamp; Offset updateDelta = null; switch (this.dragStartBehavior) { case DragStartBehavior.start: this._initialPosition = this._initialPosition + delta; updateDelta = Offset.zero; break; case DragStartBehavior.down: updateDelta = this._getDeltaForDetails(delta); break; } D.assert(updateDelta != null); this._pendingDragOffset = Offset.zero; this._lastPendingEventTimestamp = default(TimeSpan); if (this.onStart != null) { this.invokeCallback("onStart", () => { this.onStart(new DragStartDetails( sourceTimeStamp: timestamp, globalPosition: this._initialPosition )); return null; }); } if (updateDelta != Offset.zero && this.onUpdate != null) { this.invokeCallback("onUpdate", () => { this.onUpdate(new DragUpdateDetails( sourceTimeStamp: timestamp, delta: updateDelta, primaryDelta: this._getPrimaryValueFromOffset(updateDelta), globalPosition: this._initialPosition + updateDelta )); return null; }); } } } public override void rejectGesture(int pointer) { this.stopTrackingPointer(pointer); } protected override void didStopTrackingLastScrollerPointer(int pointer) { this._state = _DragState.ready; this.invokeCallback("onEnd", () => { this.onEnd(new DragEndDetails( velocity: Velocity.zero, primaryVelocity: 0.0f )); return null; }, debugReport: () => { return "Pointer scroll end"; } ); } protected override void didStopTrackingLastPointer(int pointer) { if (this._state == _DragState.possible) { this.resolve(GestureDisposition.rejected); this._state = _DragState.ready; if (this.onCancel != null) { this.invokeCallback("onCancel", () => { this.onCancel(); return null; }); } return; } bool wasAccepted = this._state == _DragState.accepted; this._state = _DragState.ready; if (wasAccepted && this.onEnd != null) { var tracker = this._velocityTrackers[pointer]; D.assert(tracker != null); var estimate = tracker.getVelocityEstimate(); if (estimate != null && this._isFlingGesture(estimate)) { Velocity velocity = new Velocity(pixelsPerSecond: estimate.pixelsPerSecond) .clampMagnitude(this.minFlingVelocity ?? Constants.kMinFlingVelocity, this.maxFlingVelocity ?? Constants.kMaxFlingVelocity); this.invokeCallback("onEnd", () => { this.onEnd(new DragEndDetails( velocity: velocity, primaryVelocity: this._getPrimaryValueFromOffset(velocity.pixelsPerSecond) )); return null; }, debugReport: () => $"{estimate}; fling at {velocity}."); } else { this.invokeCallback("onEnd", () => { this.onEnd(new DragEndDetails( velocity: Velocity.zero, primaryVelocity: 0.0f )); return null; }, debugReport: () => estimate == null ? "Could not estimate velocity." : estimate + "; judged to not be a fling." ); } } this._velocityTrackers.Clear(); } public override void dispose() { this._velocityTrackers.Clear(); base.dispose(); } public override void debugFillProperties(DiagnosticPropertiesBuilder properties) { base.debugFillProperties(properties); properties.add(new EnumProperty("start behavior", this.dragStartBehavior)); } } public class VerticalDragGestureRecognizer : DragGestureRecognizer { public VerticalDragGestureRecognizer(object debugOwner = null, PointerDeviceKind? kind = null) : base(debugOwner: debugOwner, kind: kind) { } protected override bool _isFlingGesture(VelocityEstimate estimate) { float minVelocity = this.minFlingVelocity ?? Constants.kMinFlingVelocity; float minDistance = this.minFlingDistance ?? Constants.kTouchSlop; return Mathf.Abs(estimate.pixelsPerSecond.dy) > minVelocity && Mathf.Abs(estimate.offset.dy) > minDistance; } protected override bool _hasSufficientPendingDragDeltaToAccept { get { return Mathf.Abs(this._pendingDragOffset.dy) > Constants.kTouchSlop; } } protected override Offset _getDeltaForDetails(Offset delta) { return new Offset(0.0f, delta.dy); } protected override float? _getPrimaryValueFromOffset(Offset value) { return value.dy; } public override string debugDescription { get { return "vertical drag"; } } } public class HorizontalDragGestureRecognizer : DragGestureRecognizer { public HorizontalDragGestureRecognizer(object debugOwner = null, PointerDeviceKind? kind = null) : base(debugOwner: debugOwner, kind: kind) { } protected override bool _isFlingGesture(VelocityEstimate estimate) { float minVelocity = this.minFlingVelocity ?? Constants.kMinFlingVelocity; float minDistance = this.minFlingDistance ?? Constants.kTouchSlop; return Mathf.Abs(estimate.pixelsPerSecond.dx) > minVelocity && Mathf.Abs(estimate.offset.dx) > minDistance; } protected override bool _hasSufficientPendingDragDeltaToAccept { get { return Mathf.Abs(this._pendingDragOffset.dx) > Constants.kTouchSlop; } } protected override Offset _getDeltaForDetails(Offset delta) { return new Offset(delta.dx, 0.0f); } protected override float? _getPrimaryValueFromOffset(Offset value) { return value.dx; } public override string debugDescription { get { return "horizontal drag"; } } } public class PanGestureRecognizer : DragGestureRecognizer { public PanGestureRecognizer(object debugOwner = null, PointerDeviceKind? kind = null) : base(debugOwner: debugOwner, kind: kind) { } protected override bool _isFlingGesture(VelocityEstimate estimate) { float minVelocity = this.minFlingVelocity ?? Constants.kMinFlingVelocity; float minDistance = this.minFlingDistance ?? Constants.kTouchSlop; return estimate.pixelsPerSecond.distanceSquared > minVelocity * minVelocity && estimate.offset.distanceSquared > minDistance * minDistance; } protected override bool _hasSufficientPendingDragDeltaToAccept { get { return this._pendingDragOffset.distance > Constants.kPanSlop; } } protected override Offset _getDeltaForDetails(Offset delta) { return delta; } protected override float? _getPrimaryValueFromOffset(Offset value) { return null; } public override string debugDescription { get { return "pan"; } } } }