bevy_math::common_traits

Trait StableInterpolate

source
pub trait StableInterpolate: Clone {
    // Required method
    fn interpolate_stable(&self, other: &Self, t: f32) -> Self;

    // Provided methods
    fn interpolate_stable_assign(&mut self, other: &Self, t: f32) { ... }
    fn smooth_nudge(&mut self, target: &Self, decay_rate: f32, delta: f32) { ... }
}
Expand description

A type with a natural interpolation that provides strong subdivision guarantees.

Although the only required method is interpolate_stable, many things are expected of it:

  1. The notion of interpolation should follow naturally from the semantics of the type, so that inferring the interpolation mode from the type alone is sensible.

  2. The interpolation recovers something equivalent to the starting value at t = 0.0 and likewise with the ending value at t = 1.0. They do not have to be data-identical, but they should be semantically identical. For example, Quat::slerp doesn’t always yield its second rotation input exactly at t = 1.0, but it always returns an equivalent rotation.

  3. Importantly, the interpolation must be subdivision-stable: for any interpolation curve between two (unnamed) values and any parameter-value pairs (t0, p) and (t1, q), the interpolation curve between p and q must be the linear reparametrization of the original interpolation curve restricted to the interval [t0, t1].

The last of these conditions is very strong and indicates something like constant speed. It is called “subdivision stability” because it guarantees that breaking up the interpolation into segments and joining them back together has no effect.

Here is a diagram depicting it:

top curve = u.interpolate_stable(v, t)

             t0 => p   t1 => q    
  |-------------|---------|-------------|
0 => u         /           \          1 => v
             /               \
           /                   \
         /        linear         \
       /     reparametrization     \
     /   t = t0 * (1 - s) + t1 * s   \
   /                                   \
  |-------------------------------------|
0 => p                                1 => q

bottom curve = p.interpolate_stable(q, s)

Note that some common forms of interpolation do not satisfy this criterion. For example, Quat::lerp and Rot2::nlerp are not subdivision-stable.

Furthermore, this is not to be used as a general trait for abstract interpolation. Consumers rely on the strong guarantees in order for behavior based on this trait to be well-behaved.

Required Methods§

source

fn interpolate_stable(&self, other: &Self, t: f32) -> Self

Interpolate between this value and the other given value using the parameter t. At t = 0.0, a value equivalent to self is recovered, while t = 1.0 recovers a value equivalent to other, with intermediate values interpolating between the two. See the trait-level documentation for details.

Provided Methods§

source

fn interpolate_stable_assign(&mut self, other: &Self, t: f32)

A version of interpolate_stable that assigns the result to self for convenience.

source

fn smooth_nudge(&mut self, target: &Self, decay_rate: f32, delta: f32)

Smoothly nudge this value towards the target at a given decay rate. The decay_rate parameter controls how fast the distance between self and target decays relative to the units of delta; the intended usage is for decay_rate to generally remain fixed, while delta is something like delta_time from an updating system. This produces a smooth following of the target that is independent of framerate.

More specifically, when this is called repeatedly, the result is that the distance between self and a fixed target attenuates exponentially, with the rate of this exponential decay given by decay_rate.

For example, at decay_rate = 0.0, this has no effect. At decay_rate = f32::INFINITY, self immediately snaps to target. In general, higher rates mean that self moves more quickly towards target.

§Example
let mut object_position: Vec3 = Vec3::ZERO;
let target_position: Vec3 = Vec3::new(2.0, 3.0, 5.0);
// Decay rate of ln(10) => after 1 second, remaining distance is 1/10th
let decay_rate = f32::ln(10.0);
// Calling this repeatedly will move `object_position` towards `target_position`:
object_position.smooth_nudge(&target_position, decay_rate, delta_time);

Object Safety§

This trait is not object safe.

Implementations on Foreign Types§

source§

impl<T0: StableInterpolate, T1: StableInterpolate> StableInterpolate for (T0, T1)

source§

fn interpolate_stable(&self, other: &Self, t: f32) -> Self

source§

impl<T0: StableInterpolate, T1: StableInterpolate, T2: StableInterpolate> StableInterpolate for (T0, T1, T2)

source§

fn interpolate_stable(&self, other: &Self, t: f32) -> Self

source§

impl<T0: StableInterpolate, T1: StableInterpolate, T2: StableInterpolate, T3: StableInterpolate> StableInterpolate for (T0, T1, T2, T3)

source§

fn interpolate_stable(&self, other: &Self, t: f32) -> Self

source§

impl<T0: StableInterpolate, T1: StableInterpolate, T2: StableInterpolate, T3: StableInterpolate, T4: StableInterpolate> StableInterpolate for (T0, T1, T2, T3, T4)

source§

fn interpolate_stable(&self, other: &Self, t: f32) -> Self

source§

impl<T0: StableInterpolate, T1: StableInterpolate, T2: StableInterpolate, T3: StableInterpolate, T4: StableInterpolate, T5: StableInterpolate> StableInterpolate for (T0, T1, T2, T3, T4, T5)

source§

fn interpolate_stable(&self, other: &Self, t: f32) -> Self

source§

impl<T0: StableInterpolate, T1: StableInterpolate, T2: StableInterpolate, T3: StableInterpolate, T4: StableInterpolate, T5: StableInterpolate, T6: StableInterpolate> StableInterpolate for (T0, T1, T2, T3, T4, T5, T6)

source§

fn interpolate_stable(&self, other: &Self, t: f32) -> Self

source§

impl<T0: StableInterpolate, T1: StableInterpolate, T2: StableInterpolate, T3: StableInterpolate, T4: StableInterpolate, T5: StableInterpolate, T6: StableInterpolate, T7: StableInterpolate> StableInterpolate for (T0, T1, T2, T3, T4, T5, T6, T7)

source§

fn interpolate_stable(&self, other: &Self, t: f32) -> Self

source§

impl<T0: StableInterpolate, T1: StableInterpolate, T2: StableInterpolate, T3: StableInterpolate, T4: StableInterpolate, T5: StableInterpolate, T6: StableInterpolate, T7: StableInterpolate, T8: StableInterpolate> StableInterpolate for (T0, T1, T2, T3, T4, T5, T6, T7, T8)

source§

fn interpolate_stable(&self, other: &Self, t: f32) -> Self

source§

impl<T0: StableInterpolate, T1: StableInterpolate, T2: StableInterpolate, T3: StableInterpolate, T4: StableInterpolate, T5: StableInterpolate, T6: StableInterpolate, T7: StableInterpolate, T8: StableInterpolate, T9: StableInterpolate> StableInterpolate for (T0, T1, T2, T3, T4, T5, T6, T7, T8, T9)

source§

fn interpolate_stable(&self, other: &Self, t: f32) -> Self

source§

impl<T0: StableInterpolate, T1: StableInterpolate, T2: StableInterpolate, T3: StableInterpolate, T4: StableInterpolate, T5: StableInterpolate, T6: StableInterpolate, T7: StableInterpolate, T8: StableInterpolate, T9: StableInterpolate, T10: StableInterpolate> StableInterpolate for (T0, T1, T2, T3, T4, T5, T6, T7, T8, T9, T10)

source§

fn interpolate_stable(&self, other: &Self, t: f32) -> Self

source§

impl<T: StableInterpolate> StableInterpolate for (T,)

source§

fn interpolate_stable(&self, other: &Self, t: f32) -> Self

Implementors§