use get_input_index;
use num::cast;
use primitive::InterpolationPrimitive;
pub fn spherical_linear_interpolate<T>(
input: f32,
inputs: &[f32],
outputs: &[T],
normalize: bool,
) -> T
where
T: InterpolationPrimitive + Clone,
{
let input_index = match get_input_index(input, inputs) {
Some(index) => index,
None => return outputs[0].clone(),
};
if input_index >= (inputs.len() - 1) {
outputs[outputs.len() - 1].clone()
} else {
let d = (input - inputs[input_index]) / (inputs[input_index + 1] - inputs[input_index]);
let left = &outputs[input_index];
let mut right = outputs[input_index + 1].clone();
let mut dot = left.dot(&right);
if dot < 0. {
dot = -dot;
right = right.mul(-1.);
}
let dot_threshold = cast(0.9995f32).unwrap();
let v = if dot > dot_threshold {
left.add(&right.sub(&left).mul(d))
} else {
let r_dot = if dot > 1. {
1.
} else if dot < -1. {
-1.
} else {
dot
};
let theta = r_dot.acos();
let scale1 = (theta * (1. - d)).sin();
let scale2 = (theta * d).sin();
left.mul(scale1)
.add(&right.mul(scale2))
.mul(theta.sin().recip())
};
if normalize {
v.normalize()
} else {
v
}
}
}
#[cfg(test)]
mod tests {
use super::*;
use mint::{Quaternion, Vector3};
use std;
#[test]
fn test_linear_arr3() {
let input = vec![0., 1., 2., 3., 4.];
let output = vec![
[0., 0., 0.],
[1., 0., 0.],
[0., 0., 0.],
[-1., 0., 0.],
[0., 0., 0.],
];
assert_eq!(
[std::f32::consts::FRAC_1_SQRT_2, 0., 0.],
spherical_linear_interpolate(0.5, &input, &output, false)
);
}
#[test]
fn test_linear_arr4() {
let input = vec![0., 1., 2., 3., 4.];
let output = vec![
[0., 0., 0., 0.],
[1., 0., 0., 0.],
[0., 0., 0., 0.],
[-1., 0., 0., 0.],
[0., 0., 0., 0.],
];
assert_eq!(
[0.99999994, 0., 0., 0.],
spherical_linear_interpolate(0.5, &input, &output, true)
);
}
#[test]
fn test_linear_vec3() {
let input = vec![0., 1., 2., 3., 4.];
let output = vec![
Vector3::from([0., 0., 0.]),
Vector3::from([1., 0., 0.]),
Vector3::from([0., 0., 0.]),
Vector3::from([-1., 0., 0.]),
Vector3::from([0., 0., 0.]),
];
assert_eq!(
Vector3::from([std::f32::consts::FRAC_1_SQRT_2, 0., 0.]),
spherical_linear_interpolate(0.5, &input, &output, false)
);
}
#[test]
fn test_linear_quat() {
let input = vec![0., 1., 2., 3., 4.];
let output = vec![
Quaternion::from([0., 0., 0., 0.]),
Quaternion::from([1., 0., 0., 0.]),
Quaternion::from([0., 0., 0., 0.]),
Quaternion::from([-1., 0., 0., 0.]),
Quaternion::from([0., 0., 0., 0.]),
];
assert_eq!(
Quaternion::from([0.99999994, 0., 0., 0.]),
spherical_linear_interpolate(0.5, &input, &output, true)
);
}
}