您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
417 行
15 KiB
417 行
15 KiB
using System;
|
|
using UIWidgets.foundation;
|
|
using UIWidgets.physics;
|
|
using UIWidgets.scheduler;
|
|
using UIWidgets.ui;
|
|
|
|
namespace UIWidgets.animation {
|
|
enum _AnimationDirection {
|
|
forward,
|
|
reverse,
|
|
}
|
|
|
|
public class AnimationController :
|
|
AnimationLocalStatusListenersMixinAnimationLocalListenersMixinAnimationEagerListenerMixinAnimation<double> {
|
|
public AnimationController(
|
|
double? value = null,
|
|
TimeSpan? duration = null,
|
|
string debugLabel = null,
|
|
double lowerBound = 0.0,
|
|
double upperBound = 1.0,
|
|
TickerProvider vsync = null
|
|
) {
|
|
D.assert(upperBound >= lowerBound);
|
|
D.assert(vsync != null);
|
|
this._direction = _AnimationDirection.forward;
|
|
|
|
this.duration = duration;
|
|
this.debugLabel = debugLabel;
|
|
this.lowerBound = lowerBound;
|
|
this.upperBound = upperBound;
|
|
|
|
this._ticker = vsync.createTicker(this._tick);
|
|
this._internalSetValue(value ?? lowerBound);
|
|
}
|
|
|
|
private AnimationController(
|
|
double value = 0.0,
|
|
TimeSpan? duration = null,
|
|
string debugLabel = null,
|
|
TickerProvider vsync = null
|
|
) {
|
|
D.assert(vsync != null);
|
|
this.lowerBound = double.NegativeInfinity;
|
|
this.upperBound = double.PositiveInfinity;
|
|
this._direction = _AnimationDirection.forward;
|
|
|
|
this.duration = duration;
|
|
this.debugLabel = debugLabel;
|
|
|
|
this._ticker = vsync.createTicker(this._tick);
|
|
this._internalSetValue(value);
|
|
}
|
|
|
|
public static AnimationController unbounded(
|
|
double value = 0.0,
|
|
TimeSpan? duration = null,
|
|
string debugLabel = null,
|
|
TickerProvider vsync = null
|
|
) {
|
|
return new AnimationController(value, duration, debugLabel, vsync);
|
|
}
|
|
|
|
public readonly double lowerBound;
|
|
|
|
public readonly double upperBound;
|
|
|
|
public readonly string debugLabel;
|
|
|
|
public Animation<double> view {
|
|
get { return this; }
|
|
}
|
|
|
|
public TimeSpan? duration;
|
|
|
|
Ticker _ticker;
|
|
|
|
public void resync(TickerProvider vsync) {
|
|
Ticker oldTicker = this._ticker;
|
|
this._ticker = vsync.createTicker(_tick);
|
|
this._ticker.absorbTicker(oldTicker);
|
|
}
|
|
|
|
Simulation _simulation;
|
|
|
|
public override double value {
|
|
get { return this._value; }
|
|
}
|
|
|
|
double _value;
|
|
|
|
public void setValue(double newValue) {
|
|
this.stop();
|
|
this._internalSetValue(newValue);
|
|
this.notifyListeners();
|
|
this._checkStatusChanged();
|
|
}
|
|
|
|
|
|
public void reset() {
|
|
this.setValue(this.lowerBound);
|
|
}
|
|
|
|
public double velocity {
|
|
get {
|
|
if (!this.isAnimating) {
|
|
return 0.0;
|
|
}
|
|
|
|
return this._simulation.dx((double) this.lastElapsedDuration.Value.Ticks / TimeSpan.TicksPerSecond);
|
|
}
|
|
}
|
|
|
|
void _internalSetValue(double newValue) {
|
|
this._value = newValue.clamp(this.lowerBound, this.upperBound);
|
|
if (this._value == this.lowerBound) {
|
|
this._status = AnimationStatus.dismissed;
|
|
} else if (this._value == this.upperBound) {
|
|
this._status = AnimationStatus.completed;
|
|
} else {
|
|
this._status = (this._direction == _AnimationDirection.forward)
|
|
? AnimationStatus.forward
|
|
: AnimationStatus.reverse;
|
|
}
|
|
}
|
|
|
|
TimeSpan? lastElapsedDuration {
|
|
get { return this._lastElapsedDuration; }
|
|
}
|
|
|
|
TimeSpan? _lastElapsedDuration;
|
|
|
|
public bool isAnimating {
|
|
get { return this._ticker != null && this._ticker.isActive; }
|
|
}
|
|
|
|
_AnimationDirection _direction;
|
|
|
|
public override AnimationStatus status {
|
|
get { return this._status; }
|
|
}
|
|
|
|
AnimationStatus _status;
|
|
|
|
public TickerFuture forward(double? from = null) {
|
|
D.assert(() => {
|
|
if (this.duration == null) {
|
|
throw new UIWidgetsError(
|
|
"AnimationController.forward() called with no default Duration.\n" +
|
|
"The \"duration\" property should be set, either in the constructor or later, before " +
|
|
"calling the forward() function."
|
|
);
|
|
}
|
|
|
|
return true;
|
|
});
|
|
this._direction = _AnimationDirection.forward;
|
|
if (from != null) {
|
|
this.setValue(from.Value);
|
|
}
|
|
|
|
return this._animateToInternal(this.upperBound);
|
|
}
|
|
|
|
public TickerFuture reverse(double? from = null) {
|
|
D.assert(() => {
|
|
if (this.duration == null) {
|
|
throw new UIWidgetsError(
|
|
"AnimationController.reverse() called with no default Duration.\n" +
|
|
"The \"duration\" property should be set, either in the constructor or later, before " +
|
|
"calling the reverse() function."
|
|
);
|
|
}
|
|
|
|
return true;
|
|
});
|
|
this._direction = _AnimationDirection.reverse;
|
|
if (from != null) {
|
|
this.setValue(from.Value);
|
|
}
|
|
|
|
return this._animateToInternal(this.lowerBound);
|
|
}
|
|
|
|
public TickerFuture animateTo(double target, TimeSpan? duration = null, Curve curve = null) {
|
|
curve = curve ?? Curves.linear;
|
|
|
|
this._direction = _AnimationDirection.forward;
|
|
return this._animateToInternal(target, duration: duration, curve: curve);
|
|
}
|
|
|
|
TickerFuture _animateToInternal(double target, TimeSpan? duration = null, Curve curve = null) {
|
|
curve = curve ?? Curves.linear;
|
|
|
|
TimeSpan? simulationDuration = duration;
|
|
if (simulationDuration == null) {
|
|
D.assert(() => {
|
|
if (this.duration == null) {
|
|
throw new UIWidgetsError(
|
|
"AnimationController.animateTo() called with no explicit Duration and no default Duration.\n" +
|
|
"Either the \"duration\" argument to the animateTo() method should be provided, or the " +
|
|
"\"duration\" property should be set, either in the constructor or later, before " +
|
|
"calling the animateTo() function."
|
|
);
|
|
}
|
|
|
|
return true;
|
|
});
|
|
double range = this.upperBound - this.lowerBound;
|
|
double remainingFraction = range.isFinite() ? (target - this._value).abs() / range : 1.0;
|
|
simulationDuration = TimeSpan.FromTicks((long) (this.duration.Value.Ticks * remainingFraction));
|
|
} else if (target == this.value) {
|
|
simulationDuration = TimeSpan.Zero;
|
|
}
|
|
|
|
this.stop();
|
|
|
|
if (simulationDuration == TimeSpan.Zero) {
|
|
if (this._value != target) {
|
|
this._value = target.clamp(this.lowerBound, this.upperBound);
|
|
this.notifyListeners();
|
|
}
|
|
|
|
this._status = (this._direction == _AnimationDirection.forward)
|
|
? AnimationStatus.completed
|
|
: AnimationStatus.dismissed;
|
|
this._checkStatusChanged();
|
|
return TickerFutureImpl.complete();
|
|
}
|
|
|
|
D.assert(simulationDuration > TimeSpan.Zero);
|
|
D.assert(!this.isAnimating);
|
|
return this._startSimulation(
|
|
new _InterpolationSimulation(this._value, target, simulationDuration.Value, curve));
|
|
}
|
|
|
|
public TickerFuture repeat(double? min = null, double? max = null, TimeSpan? period = null) {
|
|
min = min ?? this.lowerBound;
|
|
max = max ?? this.upperBound;
|
|
period = period ?? this.duration;
|
|
D.assert(() => {
|
|
if (period == null) {
|
|
throw new UIWidgetsError(
|
|
"AnimationController.repeat() called without an explicit period and with no default Duration.\n" +
|
|
"Either the \"period\" argument to the repeat() method should be provided, or the " +
|
|
"\"duration\" property should be set, either in the constructor or later, before " +
|
|
"calling the repeat() function."
|
|
);
|
|
}
|
|
|
|
return true;
|
|
});
|
|
return this.animateWith(new _RepeatingSimulation(min.Value, max.Value, period.Value));
|
|
}
|
|
|
|
public TickerFuture fling(double velocity = 1.0) {
|
|
this._direction = velocity < 0.0 ? _AnimationDirection.reverse : _AnimationDirection.forward;
|
|
double target = velocity < 0.0
|
|
? this.lowerBound - _kFlingTolerance.distance
|
|
: this.upperBound + _kFlingTolerance.distance;
|
|
Simulation simulation = new SpringSimulation(_kFlingSpringDescription, this.value, target, velocity);
|
|
simulation.tolerance = _kFlingTolerance;
|
|
return this.animateWith(simulation);
|
|
}
|
|
|
|
|
|
public TickerFuture animateWith(Simulation simulation) {
|
|
this.stop();
|
|
return this._startSimulation(simulation);
|
|
}
|
|
|
|
TickerFuture _startSimulation(Simulation simulation) {
|
|
D.assert(simulation != null);
|
|
D.assert(!this.isAnimating);
|
|
this._simulation = simulation;
|
|
this._lastElapsedDuration = TimeSpan.Zero;
|
|
this._value = simulation.x(0.0).clamp(this.lowerBound, this.upperBound);
|
|
var result = this._ticker.start();
|
|
this._status = (this._direction == _AnimationDirection.forward)
|
|
? AnimationStatus.forward
|
|
: AnimationStatus.reverse;
|
|
this._checkStatusChanged();
|
|
return result;
|
|
}
|
|
|
|
public void stop(bool canceled = true) {
|
|
this._simulation = null;
|
|
this._lastElapsedDuration = null;
|
|
this._ticker.stop(canceled: canceled);
|
|
}
|
|
|
|
public override void dispose() {
|
|
D.assert(() => {
|
|
if (this._ticker == null) {
|
|
throw new UIWidgetsError(
|
|
"AnimationController.dispose() called more than once.\n" +
|
|
"A given " + this.GetType() + " cannot be disposed more than once.\n" +
|
|
"The following " + this.GetType() + " object was disposed multiple times:\n" +
|
|
" " + this);
|
|
}
|
|
|
|
return true;
|
|
});
|
|
this._ticker.dispose();
|
|
this._ticker = null;
|
|
base.dispose();
|
|
}
|
|
|
|
AnimationStatus _lastReportedStatus = AnimationStatus.dismissed;
|
|
|
|
void _checkStatusChanged() {
|
|
AnimationStatus newStatus = this.status;
|
|
if (this._lastReportedStatus != newStatus) {
|
|
this._lastReportedStatus = newStatus;
|
|
this.notifyStatusListeners(newStatus);
|
|
}
|
|
}
|
|
|
|
void _tick(TimeSpan elapsed) {
|
|
this._lastElapsedDuration = elapsed;
|
|
double elapsedInSeconds = (double) elapsed.Ticks / TimeSpan.TicksPerSecond;
|
|
D.assert(elapsedInSeconds >= 0.0);
|
|
this._value = this._simulation.x(elapsedInSeconds).clamp(this.lowerBound, this.upperBound);
|
|
if (this._simulation.isDone(elapsedInSeconds)) {
|
|
this._status = (this._direction == _AnimationDirection.forward)
|
|
? AnimationStatus.completed
|
|
: AnimationStatus.dismissed;
|
|
this.stop(canceled: false);
|
|
}
|
|
|
|
this.notifyListeners();
|
|
this._checkStatusChanged();
|
|
}
|
|
|
|
public override string toStringDetails() {
|
|
string paused = this.isAnimating ? "" : "; paused";
|
|
string ticker = this._ticker == null ? "; DISPOSED" : (this._ticker.muted ? "; silenced" : "");
|
|
string label = this.debugLabel == null ? "" : "; for " + this.debugLabel;
|
|
string more = string.Format("{0} {1:F3}", base.toStringDetails(), this.value);
|
|
return more + paused + ticker + label;
|
|
}
|
|
|
|
static readonly SpringDescription _kFlingSpringDescription = SpringDescription.withDampingRatio(
|
|
mass: 1.0,
|
|
stiffness: 500.0,
|
|
ratio: 1.0
|
|
);
|
|
|
|
static readonly Tolerance _kFlingTolerance = new Tolerance(
|
|
velocity: double.PositiveInfinity,
|
|
distance: 0.01
|
|
);
|
|
}
|
|
|
|
|
|
class _InterpolationSimulation : Simulation {
|
|
internal _InterpolationSimulation(double _begin, double _end, TimeSpan duration, Curve _curve) {
|
|
this._begin = _begin;
|
|
this._end = _end;
|
|
this._curve = _curve;
|
|
|
|
D.assert(duration.Ticks > 0);
|
|
this._durationInSeconds = (double) duration.Ticks / TimeSpan.TicksPerSecond;
|
|
}
|
|
|
|
readonly double _durationInSeconds;
|
|
readonly double _begin;
|
|
readonly double _end;
|
|
readonly Curve _curve;
|
|
|
|
public override double x(double timeInSeconds) {
|
|
double t = (timeInSeconds / this._durationInSeconds).clamp(0.0, 1.0);
|
|
if (t == 0.0) {
|
|
return this._begin;
|
|
} else if (t == 1.0) {
|
|
return this._end;
|
|
} else {
|
|
return this._begin + (this._end - this._begin) * this._curve.transform(t);
|
|
}
|
|
}
|
|
|
|
public override double dx(double timeInSeconds) {
|
|
double epsilon = this.tolerance.time;
|
|
return (this.x(timeInSeconds + epsilon) - this.x(timeInSeconds - epsilon)) / (2 * epsilon);
|
|
}
|
|
|
|
public override bool isDone(double timeInSeconds) {
|
|
return timeInSeconds > this._durationInSeconds;
|
|
}
|
|
}
|
|
|
|
class _RepeatingSimulation : Simulation {
|
|
internal _RepeatingSimulation(double min, double max, TimeSpan period) {
|
|
this.min = min;
|
|
this.max = max;
|
|
this._periodInSeconds = (double) period.Ticks / TimeSpan.TicksPerSecond;
|
|
D.assert(this._periodInSeconds > 0.0);
|
|
}
|
|
|
|
readonly double min;
|
|
readonly double max;
|
|
readonly double _periodInSeconds;
|
|
|
|
public override double x(double timeInSeconds) {
|
|
D.assert(timeInSeconds >= 0.0);
|
|
double t = (timeInSeconds / this._periodInSeconds) % 1.0;
|
|
return MathUtils.lerpDouble(this.min, this.max, t);
|
|
}
|
|
|
|
public override double dx(double timeInSeconds) {
|
|
return (this.max - this.min) / this._periodInSeconds;
|
|
}
|
|
|
|
public override bool isDone(double timeInSeconds) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|