您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
390 行
14 KiB
390 行
14 KiB
using System;
|
|
using System.Collections.Generic;
|
|
using Unity.UIWidgets.async;
|
|
using Unity.UIWidgets.foundation;
|
|
using Unity.UIWidgets.ui;
|
|
using UnityEngine;
|
|
|
|
namespace Unity.UIWidgets.gestures {
|
|
public delegate T RecognizerCallback<T>();
|
|
|
|
public enum DragStartBehavior {
|
|
down,
|
|
start
|
|
}
|
|
|
|
public abstract class GestureRecognizer : DiagnosticableTree, GestureArenaMember {
|
|
protected GestureRecognizer(object debugOwner = null, PointerDeviceKind? kind = null) {
|
|
this.debugOwner = debugOwner;
|
|
this._kindFilter = kind;
|
|
}
|
|
|
|
public readonly object debugOwner;
|
|
|
|
readonly PointerDeviceKind? _kindFilter;
|
|
|
|
readonly Dictionary<int, PointerDeviceKind> _pointerToKind = new Dictionary<int, PointerDeviceKind>{};
|
|
|
|
public void addPointer(PointerDownEvent evt) {
|
|
this._pointerToKind[evt.pointer] = evt.kind;
|
|
if (this.isPointerAllowed(evt)) {
|
|
this.addAllowedPointer(evt);
|
|
}
|
|
else {
|
|
this.handleNonAllowedPointer(evt);
|
|
}
|
|
}
|
|
|
|
public abstract void addAllowedPointer(PointerDownEvent evt);
|
|
|
|
protected virtual void handleNonAllowedPointer(PointerDownEvent evt) {
|
|
}
|
|
|
|
protected virtual bool isPointerAllowed(PointerDownEvent evt) {
|
|
return this._kindFilter == null || this._kindFilter == evt.kind;
|
|
}
|
|
|
|
protected virtual PointerDeviceKind getKindForPointer(int pointer) {
|
|
D.assert(this._pointerToKind.ContainsKey(pointer));
|
|
return this._pointerToKind[pointer];
|
|
}
|
|
|
|
public virtual void addScrollPointer(PointerScrollEvent evt) {
|
|
}
|
|
|
|
public virtual void dispose() {
|
|
}
|
|
|
|
public abstract string debugDescription { get; }
|
|
|
|
protected T invokeCallback<T>(string name, RecognizerCallback<T> callback, Func<string> debugReport = null) {
|
|
D.assert(callback != null);
|
|
|
|
T result = default(T);
|
|
try {
|
|
D.assert(() => {
|
|
if (D.debugPrintRecognizerCallbacksTrace) {
|
|
var report = debugReport != null ? debugReport() : null;
|
|
// The 19 in the line below is the width of the prefix used by
|
|
// _debugLogDiagnostic in arena.dart.
|
|
var prefix = D.debugPrintGestureArenaDiagnostics ? new string(' ', 19) + "❙ " : "";
|
|
Debug.LogFormat("{0}this calling {1} callback.{2}",
|
|
prefix, name, report.isNotEmpty() ? " " + report : "");
|
|
}
|
|
|
|
return true;
|
|
});
|
|
|
|
result = callback();
|
|
}
|
|
catch (Exception ex) {
|
|
UIWidgetsError.reportError(new UIWidgetsErrorDetails(
|
|
exception: ex,
|
|
library: "gesture",
|
|
context: "while handling a gesture",
|
|
informationCollector: information => {
|
|
information.AppendLine("Handler: " + name);
|
|
information.AppendLine("Recognizer:");
|
|
information.AppendLine(" " + this);
|
|
}
|
|
));
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
public abstract void acceptGesture(int pointer);
|
|
public abstract void rejectGesture(int pointer);
|
|
|
|
public override void debugFillProperties(DiagnosticPropertiesBuilder properties) {
|
|
base.debugFillProperties(properties);
|
|
properties.add(new DiagnosticsProperty<object>("debugOwner", this.debugOwner,
|
|
defaultValue: Diagnostics.kNullDefaultValue));
|
|
}
|
|
}
|
|
|
|
public abstract class OneSequenceGestureRecognizer : GestureRecognizer {
|
|
protected OneSequenceGestureRecognizer(object debugOwner = null, PointerDeviceKind? kind = null) : base(
|
|
debugOwner, kind) {
|
|
}
|
|
|
|
readonly Dictionary<int, GestureArenaEntry> _entries = new Dictionary<int, GestureArenaEntry>();
|
|
|
|
readonly HashSet<int> _trackedPointers = new HashSet<int>();
|
|
|
|
protected abstract void handleEvent(PointerEvent evt);
|
|
|
|
public override void acceptGesture(int pointer) {
|
|
}
|
|
|
|
public override void rejectGesture(int pointer) {
|
|
}
|
|
|
|
protected abstract void didStopTrackingLastPointer(int pointer);
|
|
|
|
protected virtual void didStopTrackingLastScrollerPointer(int pointer) {
|
|
}
|
|
|
|
protected virtual void resolve(GestureDisposition disposition) {
|
|
var localEntries = new List<GestureArenaEntry>(this._entries.Values);
|
|
this._entries.Clear();
|
|
foreach (GestureArenaEntry entry in localEntries) {
|
|
entry.resolve(disposition);
|
|
}
|
|
}
|
|
|
|
protected virtual void resolvePointer(int pointer, GestureDisposition disposition) {
|
|
GestureArenaEntry entry = _entries[pointer];
|
|
if (entry != null) {
|
|
entry.resolve(disposition);
|
|
_entries.Remove(pointer);
|
|
}
|
|
}
|
|
|
|
public override void dispose() {
|
|
this.resolve(GestureDisposition.rejected);
|
|
foreach (int pointer in this._trackedPointers) {
|
|
GestureBinding.instance.pointerRouter.removeRoute(pointer, this.handleEvent);
|
|
}
|
|
|
|
this._trackedPointers.Clear();
|
|
D.assert(this._entries.isEmpty());
|
|
base.dispose();
|
|
}
|
|
|
|
public GestureArenaTeam team {
|
|
get { return this._team; }
|
|
set {
|
|
D.assert(value != null);
|
|
D.assert(this._entries.isEmpty());
|
|
D.assert(this._trackedPointers.isEmpty());
|
|
D.assert(this._team == null);
|
|
this._team = value;
|
|
}
|
|
}
|
|
|
|
GestureArenaTeam _team;
|
|
|
|
GestureArenaEntry _addPointerToArena(int pointer) {
|
|
if (this._team != null) {
|
|
return this._team.add(pointer, this);
|
|
}
|
|
|
|
return GestureBinding.instance.gestureArena.add(pointer, this);
|
|
}
|
|
|
|
protected void startTrackingScrollerPointer(int pointer) {
|
|
GestureBinding.instance.pointerRouter.addRoute(pointer, this.handleEvent);
|
|
}
|
|
|
|
protected void stopTrackingScrollerPointer(int pointer) {
|
|
if (this._trackedPointers.isEmpty()) {
|
|
this.didStopTrackingLastScrollerPointer(pointer);
|
|
}
|
|
}
|
|
|
|
protected void startTrackingPointer(int pointer, Matrix4 transform = null) {
|
|
GestureBinding.instance.pointerRouter.addRoute(pointer, this.handleEvent, transform);
|
|
this._trackedPointers.Add(pointer);
|
|
D.assert(!this._entries.ContainsKey(pointer));
|
|
this._entries[pointer] = this._addPointerToArena(pointer);
|
|
}
|
|
|
|
protected void stopTrackingPointer(int pointer) {
|
|
if (this._trackedPointers.Contains(pointer)) {
|
|
GestureBinding.instance.pointerRouter.removeRoute(pointer, this.handleEvent);
|
|
this._trackedPointers.Remove(pointer);
|
|
if (this._trackedPointers.isEmpty()) {
|
|
this.didStopTrackingLastPointer(pointer);
|
|
}
|
|
}
|
|
}
|
|
|
|
protected void stopTrackingIfPointerNoLongerDown(PointerEvent evt) {
|
|
if (evt is PointerUpEvent || evt is PointerCancelEvent) {
|
|
this.stopTrackingPointer(evt.pointer);
|
|
}
|
|
}
|
|
}
|
|
|
|
public enum GestureRecognizerState {
|
|
ready,
|
|
possible,
|
|
accepted,
|
|
defunct,
|
|
}
|
|
|
|
public abstract class PrimaryPointerGestureRecognizer : OneSequenceGestureRecognizer {
|
|
protected PrimaryPointerGestureRecognizer(
|
|
TimeSpan? deadline = null,
|
|
object debugOwner = null,
|
|
PointerDeviceKind? kind = null,
|
|
float? preAcceptSlopTolerance = Constants.kTouchSlop,
|
|
float? postAcceptSlopTolerance = Constants.kTouchSlop
|
|
) : base(debugOwner: debugOwner, kind: kind) {
|
|
D.assert(preAcceptSlopTolerance == null || preAcceptSlopTolerance >= 0,
|
|
() => "The preAcceptSlopTolerance must be positive or null");
|
|
|
|
D.assert(postAcceptSlopTolerance == null || postAcceptSlopTolerance >= 0,
|
|
() => "The postAcceptSlopTolerance must be positive or null");
|
|
|
|
this.deadline = deadline;
|
|
this.preAcceptSlopTolerance = preAcceptSlopTolerance;
|
|
this.postAcceptSlopTolerance = postAcceptSlopTolerance;
|
|
}
|
|
|
|
public readonly TimeSpan? deadline;
|
|
|
|
public readonly float? preAcceptSlopTolerance;
|
|
|
|
public readonly float? postAcceptSlopTolerance;
|
|
|
|
public GestureRecognizerState state = GestureRecognizerState.ready;
|
|
|
|
public int primaryPointer;
|
|
|
|
public OffsetPair initialPosition;
|
|
|
|
Timer _timer;
|
|
|
|
public override void addAllowedPointer(PointerDownEvent evt) {
|
|
this.startTrackingPointer(evt.pointer, evt.transform);
|
|
if (this.state == GestureRecognizerState.ready) {
|
|
this.state = GestureRecognizerState.possible;
|
|
this.primaryPointer = evt.pointer;
|
|
this.initialPosition = new OffsetPair(local: evt.localPosition, global: evt.position);
|
|
if (this.deadline != null) {
|
|
this._timer = Window.instance.run(this.deadline.Value, () => didExceedDeadlineWithEvent(evt));
|
|
}
|
|
}
|
|
}
|
|
|
|
protected override void handleEvent(PointerEvent evt) {
|
|
D.assert(this.state != GestureRecognizerState.ready);
|
|
|
|
if (evt.pointer == this.primaryPointer) {
|
|
bool isPreAcceptSlopPastTolerance = this.state == GestureRecognizerState.possible &&
|
|
this.preAcceptSlopTolerance != null &&
|
|
this._getGlobalDistance(evt) > this.preAcceptSlopTolerance;
|
|
bool isPostAcceptSlopPastTolerance = this.state == GestureRecognizerState.accepted &&
|
|
this.postAcceptSlopTolerance != null &&
|
|
this._getGlobalDistance(evt) > this.postAcceptSlopTolerance;
|
|
|
|
if (evt is PointerMoveEvent && (isPreAcceptSlopPastTolerance || isPostAcceptSlopPastTolerance)) {
|
|
this.resolve(GestureDisposition.rejected);
|
|
this.stopTrackingPointer(this.primaryPointer);
|
|
}
|
|
else {
|
|
this.handlePrimaryPointer(evt);
|
|
}
|
|
}
|
|
|
|
this.stopTrackingIfPointerNoLongerDown(evt);
|
|
}
|
|
|
|
protected abstract void handlePrimaryPointer(PointerEvent evt);
|
|
|
|
protected virtual void didExceedDeadline() {
|
|
D.assert(this.deadline == null);
|
|
}
|
|
|
|
protected virtual void didExceedDeadlineWithEvent(PointerDownEvent evt) {
|
|
this.didExceedDeadline();
|
|
}
|
|
|
|
public override void acceptGesture(int pointer) {
|
|
if (pointer == this.primaryPointer && this.state == GestureRecognizerState.possible) {
|
|
this._stopTimer();
|
|
this.state = GestureRecognizerState.accepted;
|
|
}
|
|
}
|
|
|
|
public override void rejectGesture(int pointer) {
|
|
if (pointer == this.primaryPointer && (this.state == GestureRecognizerState.possible ||
|
|
this.state == GestureRecognizerState.accepted)) {
|
|
this._stopTimer();
|
|
this.state = GestureRecognizerState.defunct;
|
|
}
|
|
}
|
|
|
|
protected override void didStopTrackingLastPointer(int pointer) {
|
|
this._stopTimer();
|
|
this.state = GestureRecognizerState.ready;
|
|
}
|
|
|
|
public override void dispose() {
|
|
this._stopTimer();
|
|
base.dispose();
|
|
}
|
|
|
|
void _stopTimer() {
|
|
if (this._timer != null) {
|
|
this._timer.cancel();
|
|
this._timer = null;
|
|
}
|
|
}
|
|
|
|
float _getGlobalDistance(PointerEvent evt) {
|
|
Offset offset = evt.position - this.initialPosition.global;
|
|
return offset.distance;
|
|
}
|
|
|
|
public override void debugFillProperties(DiagnosticPropertiesBuilder properties) {
|
|
base.debugFillProperties(properties);
|
|
properties.add(new EnumProperty<GestureRecognizerState>("state", this.state));
|
|
}
|
|
}
|
|
|
|
public class OffsetPair {
|
|
/// Creates a [OffsetPair] combining a [local] and [global] [Offset].
|
|
public OffsetPair(
|
|
Offset local = null,
|
|
Offset global = null
|
|
) {
|
|
this.local = local;
|
|
this.global = global;
|
|
}
|
|
|
|
/// Creates a [OffsetPair] from [PointerEvent.localPosition] and
|
|
/// [PointerEvent.position].
|
|
public static OffsetPair fromEventPosition(PointerEvent evt) {
|
|
return new OffsetPair(local: evt.localPosition, global: evt.position);
|
|
}
|
|
|
|
/// Creates a [OffsetPair] from [PointerEvent.localDelta] and
|
|
/// [PointerEvent.delta].
|
|
public static OffsetPair fromEventDelta(PointerEvent evt) {
|
|
return new OffsetPair(local: evt.localDelta, global: evt.delta);
|
|
}
|
|
|
|
/// A [OffsetPair] where both [Offset]s are [Offset.zero].
|
|
public readonly static OffsetPair zero = new OffsetPair(local: Offset.zero, global: Offset.zero);
|
|
|
|
/// The [Offset] in the local coordinate space.
|
|
public readonly Offset local;
|
|
|
|
/// The [Offset] in the global coordinate space after conversion to logical
|
|
/// pixels.
|
|
public readonly Offset global;
|
|
|
|
/// Adds the `other.global` to [global] and `other.local` to [local].
|
|
public static OffsetPair operator +(OffsetPair current, OffsetPair other) {
|
|
return new OffsetPair(
|
|
local: current.local + other.local,
|
|
global: current.global + other.global
|
|
);
|
|
}
|
|
|
|
/// Subtracts the `other.global` from [global] and `other.local` from [local].
|
|
public static OffsetPair operator -(OffsetPair current, OffsetPair other) {
|
|
return new OffsetPair(
|
|
local: current.local - other.local,
|
|
global: current.global - other.global
|
|
);
|
|
}
|
|
|
|
public override string ToString() {
|
|
return $"runtimeType(local: ${this.local}, global: ${this.global})";
|
|
}
|
|
}
|
|
}
|