Skip to content

Commit

Permalink
Revert "Revert "Fixed rate animation loop fixes (#3396)" (#3398)"
Browse files Browse the repository at this point in the history
This reverts commit 0ab46bc.
  • Loading branch information
wdanilo authored Apr 13, 2022
1 parent 0ea5dc2 commit a514b28
Show file tree
Hide file tree
Showing 4 changed files with 220 additions and 39 deletions.
200 changes: 174 additions & 26 deletions lib/rust/ensogl/core/src/animation/loops.rs
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@ use web::Closure;
/// differ across browsers and browser versions. We have even observed that `performance.now()` can
/// sometimes provide a bigger value than time provided to `requestAnimationFrame` callback later,
/// which resulted in a negative frame time.
#[derive(Clone, Copy, Debug, Default)]
#[derive(Clone, Copy, Debug, Default, PartialEq)]
#[allow(missing_docs)]
pub struct TimeInfo {
pub animation_loop_start: Duration,
Expand All @@ -36,7 +36,15 @@ impl TimeInfo {
/// Check whether the time info was initialized. See the documentation of the struct to learn
/// more.
pub fn is_initialized(&self) -> bool {
self.animation_loop_start != 0.0.ms()
self.animation_loop_start != 0.ms()
}

/// Creates a new [`TimeInfo`] for the next frame with the provided time. The frame time will
/// be computed based on the current and the new frame time.
pub fn new_frame(mut self, since_animation_loop_started: Duration) -> Self {
self.previous_frame = since_animation_loop_started - self.since_animation_loop_started;
self.since_animation_loop_started = since_animation_loop_started;
self
}
}

Expand Down Expand Up @@ -127,6 +135,7 @@ impl<Callback> Drop for RawLoopData<Callback> {
// === Types ===

pub trait LoopCallback = FnMut(TimeInfo) + 'static;
pub trait OnTooManyFramesSkippedCallback = FnMut() + 'static;


// === Definition ===
Expand Down Expand Up @@ -183,51 +192,90 @@ where
// =============================

/// A callback `FnMut(TimeInfo) -> FnMut(TimeInfo)` transformer. Calls the inner callback with a
/// constant frame rate.
/// constant frame rate. If too many frames were skipped, the [`on_too_many_frames_skipped`]
/// callback will be used instead.
#[derive(Derivative)]
#[derivative(Debug(bound = ""))]
pub struct FixedFrameRateSampler<Callback> {
frame_time: Duration,
local_time: Duration,
time_buffer: Duration,
#[allow(missing_docs)]
pub struct FixedFrameRateSampler<Callback, OnTooManyFramesSkipped> {
pub max_skipped_frames: usize,
frame_time: Duration,
local_time: Duration,
time_buffer: Duration,
#[derivative(Debug = "ignore")]
callback: Callback,
callback: Callback,
#[derivative(Debug = "ignore")]
on_too_many_frames_skipped: OnTooManyFramesSkipped,
}

impl<Callback> FixedFrameRateSampler<Callback> {
impl<Callback, OnTooManyFramesSkipped> FixedFrameRateSampler<Callback, OnTooManyFramesSkipped> {
/// Constructor.
pub fn new(frame_rate: f32, callback: Callback) -> Self {
pub fn new(
frame_rate: f32,
callback: Callback,
on_too_many_frames_skipped: OnTooManyFramesSkipped,
) -> Self {
let max_skipped_frames = 2;
let frame_time = (1000.0 / frame_rate).ms();
let local_time = default();
let time_buffer = default();
Self { frame_time, local_time, time_buffer, callback }
// The first call to this sampler will be with frame time 0, which would drop this
// `time_buffer` to 0.
let time_buffer = frame_time;
Self {
max_skipped_frames,
frame_time,
local_time,
time_buffer,
callback,
on_too_many_frames_skipped,
}
}
}

impl<Callback: FnOnce<(TimeInfo,)>> FnOnce<(TimeInfo,)> for FixedFrameRateSampler<Callback> {
impl<Callback: FnOnce<(TimeInfo,)>, OnTooManyFramesSkipped> FnOnce<(TimeInfo,)>
for FixedFrameRateSampler<Callback, OnTooManyFramesSkipped>
{
type Output = ();
extern "rust-call" fn call_once(self, args: (TimeInfo,)) -> Self::Output {
self.callback.call_once(args);
}
}

impl<Callback: FnMut<(TimeInfo,)>> FnMut<(TimeInfo,)> for FixedFrameRateSampler<Callback> {
impl<Callback, OnTooManyFramesSkipped> FnMut<(TimeInfo,)>
for FixedFrameRateSampler<Callback, OnTooManyFramesSkipped>
where
Callback: FnMut(TimeInfo),
OnTooManyFramesSkipped: FnMut(),
{
extern "rust-call" fn call_mut(&mut self, args: (TimeInfo,)) -> Self::Output {
let time = args.0;
self.time_buffer += time.previous_frame;
loop {
if self.time_buffer < 0.0.ms() {
break;
} else {
let mut time = args.0;
self.time_buffer += time.since_animation_loop_started - self.local_time;

let frame_time_2 = self.frame_time * 0.5;
let skipped_frames = ((self.time_buffer - frame_time_2) / self.frame_time) as usize;
let too_many_frames_skipped = skipped_frames > self.max_skipped_frames;
if !too_many_frames_skipped {
for _ in 0..skipped_frames {
self.local_time += self.frame_time;
self.time_buffer -= self.frame_time;
let animation_loop_start = time.animation_loop_start;
let previous_frame = self.frame_time;
let since_animation_loop_started = self.local_time;
let time2 =
TimeInfo { animation_loop_start, previous_frame, since_animation_loop_started };
self.local_time += self.frame_time;
self.callback.call_mut((time2,));
}
let not_too_fast_refresh_rate = self.time_buffer >= -frame_time_2;
if not_too_fast_refresh_rate {
self.time_buffer -= self.frame_time;
}
time.previous_frame = time.since_animation_loop_started - self.local_time;
self.local_time = time.since_animation_loop_started;
(self.callback)(time);
} else {
self.local_time = time.since_animation_loop_started;
self.time_buffer = 0.ms();
(self.on_too_many_frames_skipped)();
}
}
}
Expand All @@ -239,13 +287,113 @@ impl<Callback: FnMut<(TimeInfo,)>> FnMut<(TimeInfo,)> for FixedFrameRateSampler<
// ==========================

/// Loop with a `FixedFrameRateSampler` attached.
pub type FixedFrameRateLoop<Callback> = Loop<FixedFrameRateSampler<Callback>>;
pub type FixedFrameRateLoop<Callback, OnTooManyFramesSkipped> =
Loop<FixedFrameRateSampler<Callback, OnTooManyFramesSkipped>>;

impl<Callback> FixedFrameRateLoop<Callback>
where Callback: LoopCallback
impl<Callback, OnTooManyFramesSkipped> FixedFrameRateLoop<Callback, OnTooManyFramesSkipped>
where
Callback: LoopCallback,
OnTooManyFramesSkipped: OnTooManyFramesSkippedCallback,
{
/// Constructor.
pub fn new_with_fixed_frame_rate(frame_rate: f32, callback: Callback) -> Self {
Self::new(FixedFrameRateSampler::new(frame_rate, callback))
pub fn new_with_fixed_frame_rate(
frame_rate: f32,
callback: Callback,
on_too_many_frames_skipped: OnTooManyFramesSkipped,
) -> Self {
Self::new(FixedFrameRateSampler::new(frame_rate, callback, on_too_many_frames_skipped))
}
}



// =============
// === Tests ===
// =============

#[cfg(test)]
mod tests {
use super::*;
use std::collections::VecDeque;

#[test]
fn fixed_frame_rate_sampler_test() {
let mut count_check = 0;
let count = Rc::new(Cell::new(0));
let too_many_frames_skipped_count = Rc::new(Cell::new(0));
let frame_times = Rc::new(RefCell::new(VecDeque::new()));
let mut lp = FixedFrameRateSampler::new(
10.0,
|t| {
frame_times.borrow_mut().push_back(t);
count.set(count.get() + 1);
},
|| {
too_many_frames_skipped_count.set(too_many_frames_skipped_count.get() + 1);
},
);
let mut time = TimeInfo {
animation_loop_start: 0.ms(),
previous_frame: 0.ms(),
since_animation_loop_started: 0.ms(),
};

let mut step = |frame_time: Duration,
sub_frames: &[Duration],
offset: Duration,
skipped_count: Option<usize>| {
let time2 = time.new_frame(frame_time);
lp(time2);
for sub_frame in sub_frames {
count_check += 1;
time = time.new_frame(*sub_frame);
assert_eq!(frame_times.borrow_mut().pop_front(), Some(time));
}
time = time.new_frame(time2.since_animation_loop_started);
if skipped_count.is_none() {
count_check += 1;
assert_eq!(frame_times.borrow_mut().pop_front(), Some(time));
}
assert_eq!(frame_times.borrow_mut().pop_front(), None);
assert_eq!(count.get(), count_check);
assert_eq!(lp.time_buffer, offset);
if let Some(skipped_count) = skipped_count {
assert_eq!(too_many_frames_skipped_count.get(), skipped_count);
}
};

// Start frame.
step(0.ms(), &[], 0.ms(), None);

// Perfectly timed next frame.
step(100.ms(), &[], 0.ms(), None);

// Skipping 2 frames.
step(400.ms(), &[200.ms(), 300.ms()], 0.ms(), None);

// Perfectly timed next frame.
step(500.ms(), &[], 0.ms(), None);

// Next frame too slow.
step(640.ms(), &[], 40.ms(), None);

// Next frame too slow.
step(800.ms(), &[740.ms()], 0.ms(), None);

// Not-perfectly timed next frames.
step(870.ms(), &[], -30.ms(), None);
step(1010.ms(), &[], 10.ms(), None);
step(1090.ms(), &[], -10.ms(), None);
step(1200.ms(), &[], 0.ms(), None);

// Next frames way too fast.
step(1210.ms(), &[], -90.ms(), None);
// Time compression – we don't want to accumulate too much of negative time buffer for
// monitors with bigger refresh-rate than assumed. The total accumulated time buffer would
// be -180 here, so we add a frame time to it (100).
step(1220.ms(), &[], -80.ms(), None);

// Too many frames skipped.
step(2000.ms(), &[], 0.ms(), Some(1));
}
}
30 changes: 26 additions & 4 deletions lib/rust/ensogl/core/src/animation/physics/inertia.rs
Original file line number Diff line number Diff line change
Expand Up @@ -707,7 +707,12 @@ where
if self.animation_loop.get().is_none() {
let frame_rate = self.frame_rate.get();
let step = step(self);
let animation_loop = animation::Loop::new_with_fixed_frame_rate(frame_rate, step);
let on_too_many_frames_skipped = on_too_many_frames_skipped(self);
let animation_loop = animation::Loop::new_with_fixed_frame_rate(
frame_rate,
step,
on_too_many_frames_skipped,
);
self.animation_loop.set(Some(animation_loop));
self.on_start.call();
}
Expand Down Expand Up @@ -778,9 +783,11 @@ impl<T, OnStep, OnStart, OnEnd> WeakAnimationLoop<T, OnStep, OnStart, OnEnd> {

// === Animation Step ===

/// Alias for `FixedFrameRateLoop` with specified step callback.
pub type FixedFrameRateAnimationStep<T, OnStep, OnStart, OnEnd> =
animation::FixedFrameRateLoop<Step<T, OnStep, OnStart, OnEnd>>;
/// Alias for [`FixedFrameRateLoop`] with specified step callback.
pub type FixedFrameRateAnimationStep<T, OnStep, OnStart, OnEnd> = animation::FixedFrameRateLoop<
Step<T, OnStep, OnStart, OnEnd>,
OnTooManyFramesSkipped<T, OnStep, OnStart, OnEnd>,
>;

/// Callback for an animation step.
pub type Step<T, OnStep, OnStart, OnEnd> = impl Fn(animation::TimeInfo);
Expand All @@ -805,6 +812,21 @@ where
}
}

/// Callback for an animation step.
pub type OnTooManyFramesSkipped<T, OnStep, OnStart, OnEnd> = impl Fn();

fn on_too_many_frames_skipped<T, OnStep, OnStart, OnEnd>(
simulator: &Simulator<T, OnStep, OnStart, OnEnd>,
) -> OnTooManyFramesSkipped<T, OnStep, OnStart, OnEnd>
where
T: Value,
OnStep: Callback1<T>,
OnStart: Callback0,
OnEnd: Callback1<EndStatus>, {
let data = simulator.data.clone_ref();
move || data.skip()
}



// =================
Expand Down
1 change: 1 addition & 0 deletions lib/rust/ensogl/core/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@
#![warn(unsafe_code)]
// === Non-Standard Linter Configuration ===
#![allow(clippy::option_map_unit_fn)]
#![allow(clippy::precedence)]
#![allow(dead_code)]
#![deny(unconditional_recursion)]
#![warn(missing_copy_implementations)]
Expand Down
28 changes: 19 additions & 9 deletions lib/rust/types/src/unit2.rs
Original file line number Diff line number Diff line change
Expand Up @@ -152,7 +152,6 @@ impl<V, R> AsRef<UnitData<V, R>> for UnitData<V, R> {
// === Eq ===
// ==========

impl<V, R: PartialEq> Eq for UnitData<V, R> {}
impl<V, R: PartialEq> PartialEq for UnitData<V, R> {
fn eq(&self, other: &Self) -> bool {
self.repr.eq(&other.repr)
Expand All @@ -165,12 +164,6 @@ impl<V, R: PartialEq> PartialEq for UnitData<V, R> {
// === Ord ===
// ===========

impl<V, R: Ord> Ord for UnitData<V, R> {
fn cmp(&self, other: &Self) -> std::cmp::Ordering {
self.repr.cmp(&other.repr)
}
}

impl<V, R: PartialOrd> PartialOrd for UnitData<V, R> {
fn partial_cmp(&self, other: &Self) -> Option<std::cmp::Ordering> {
self.repr.partial_cmp(&other.repr)
Expand Down Expand Up @@ -354,8 +347,6 @@ macro_rules! gen_ops_mut {
};
}



gen_ops!(RevAdd, Add, add);
gen_ops!(RevSub, Sub, sub);
gen_ops!(RevMul, Mul, mul);
Expand All @@ -368,6 +359,15 @@ gen_ops_mut!(RevSub, Sub, SubAssign, sub_assign);
gen_ops_mut!(RevMul, Mul, MulAssign, mul_assign);
gen_ops_mut!(RevDiv, Div, DivAssign, div_assign);

impl<V, R: ops::Neg<Output = R>> ops::Neg for UnitData<V, R> {
type Output = UnitData<V, R>;

fn neg(mut self) -> Self::Output {
self.repr = self.repr.neg();
self
}
}



// ==============================
Expand Down Expand Up @@ -618,6 +618,16 @@ impl const DurationNumberOps for f32 {
}
}

impl const DurationNumberOps for i64 {
fn ms(self) -> Duration {
(self as f32).ms()
}

fn s(self) -> Duration {
(self as f32).s()
}
}

impl From<std::time::Duration> for Duration {
fn from(duration: std::time::Duration) -> Self {
(duration.as_millis() as <DURATION as Variant>::Repr).ms()
Expand Down

0 comments on commit a514b28

Please sign in to comment.