use minterpolate::InterpolationPrimitive;
use serde::{Deserialize, Serialize};
use amethyst_core::{
alga::general::{SubsetOf, SupersetOf},
ecs::prelude::{Entity, WriteStorage},
math::{convert, RealField, Vector2, Vector3, Vector4},
};
use crate::resources::{AnimationControlSet, AnimationSampling};
use self::SamplerPrimitive::*;
pub fn get_animation_set<'a, I, T>(
controls: &'a mut WriteStorage<'_, AnimationControlSet<I, T>>,
entity: Entity,
) -> Option<&'a mut AnimationControlSet<I, T>>
where
I: Send + Sync + 'static,
T: AnimationSampling,
{
controls
.entry(entity)
.ok()
.map(|entry| entry.or_insert_with(AnimationControlSet::default))
}
#[derive(Debug, Clone, Copy, Serialize, Deserialize)]
pub enum SamplerPrimitive<S>
where
S: RealField + SubsetOf<f32> + SupersetOf<f32>,
{
Scalar(S),
Vec2([S; 2]),
Vec3([S; 3]),
Vec4([S; 4]),
}
impl<S> From<[S; 2]> for SamplerPrimitive<S>
where
S: RealField + SubsetOf<f32> + SupersetOf<f32>,
{
fn from(arr: [S; 2]) -> Self {
SamplerPrimitive::Vec2(arr)
}
}
impl<S> From<[S; 3]> for SamplerPrimitive<S>
where
S: RealField + SubsetOf<f32> + SupersetOf<f32>,
{
fn from(arr: [S; 3]) -> Self {
SamplerPrimitive::Vec3(arr)
}
}
impl<S> From<[S; 4]> for SamplerPrimitive<S>
where
S: RealField + SubsetOf<f32> + SupersetOf<f32>,
{
fn from(arr: [S; 4]) -> Self {
SamplerPrimitive::Vec4(arr)
}
}
impl<S> From<Vector2<S>> for SamplerPrimitive<S>
where
S: RealField + SubsetOf<f32> + SupersetOf<f32>,
{
fn from(arr: Vector2<S>) -> Self {
SamplerPrimitive::Vec2(arr.into())
}
}
impl<S> From<Vector3<S>> for SamplerPrimitive<S>
where
S: RealField + SubsetOf<f32> + SupersetOf<f32>,
{
fn from(arr: Vector3<S>) -> Self {
SamplerPrimitive::Vec3(arr.into())
}
}
impl<S> From<Vector4<S>> for SamplerPrimitive<S>
where
S: RealField + SubsetOf<f32> + SupersetOf<f32>,
{
fn from(arr: Vector4<S>) -> Self {
SamplerPrimitive::Vec4(arr.into())
}
}
impl<S> InterpolationPrimitive for SamplerPrimitive<S>
where
S: RealField + SubsetOf<f32> + SupersetOf<f32>,
{
fn add(&self, other: &Self) -> Self {
match (*self, *other) {
(Scalar(ref s), Scalar(ref o)) => Scalar(*s + *o),
(Vec2(ref s), Vec2(ref o)) => Vec2([s[0] + o[0], s[1] + o[1]]),
(Vec3(ref s), Vec3(ref o)) => Vec3([s[0] + o[0], s[1] + o[1], s[2] + o[2]]),
(Vec4(ref s), Vec4(ref o)) => {
Vec4([s[0] + o[0], s[1] + o[1], s[2] + o[2], s[3] + o[3]])
}
_ => panic!("Interpolation can not be done between primitives of different types"),
}
}
fn sub(&self, other: &Self) -> Self {
match (*self, *other) {
(Scalar(ref s), Scalar(ref o)) => Scalar(*s - *o),
(Vec2(ref s), Vec2(ref o)) => Vec2([s[0] - o[0], s[1] - o[1]]),
(Vec3(ref s), Vec3(ref o)) => Vec3([s[0] - o[0], s[1] - o[1], s[2] - o[2]]),
(Vec4(ref s), Vec4(ref o)) => {
Vec4([s[0] - o[0], s[1] - o[1], s[2] - o[2], s[3] - o[3]])
}
_ => panic!("Interpolation can not be done between primitives of different types"),
}
}
fn mul(&self, scalar: f32) -> Self {
match *self {
Scalar(ref s) => Scalar(mul_f32(*s, scalar)),
Vec2(ref s) => Vec2([mul_f32(s[0], scalar), mul_f32(s[1], scalar)]),
Vec3(ref s) => Vec3([
mul_f32(s[0], scalar),
mul_f32(s[1], scalar),
mul_f32(s[2], scalar),
]),
Vec4(ref s) => Vec4([
mul_f32(s[0], scalar),
mul_f32(s[1], scalar),
mul_f32(s[2], scalar),
mul_f32(s[3], scalar),
]),
}
}
fn dot(&self, other: &Self) -> f32 {
convert(match (*self, *other) {
(Scalar(s), Scalar(o)) => (s * o),
(Vec2(s), Vec2(o)) => (s[0] * o[0] + s[1] * o[1]),
(Vec3(s), Vec3(o)) => (s[0] * o[0] + s[1] * o[1] + s[2] * o[2]),
(Vec4(s), Vec4(o)) => (s[0] * o[0] + s[1] * o[1] + s[2] * o[2] + s[3] * o[3]),
_ => panic!("Interpolation can not be done between primitives of different types"),
})
}
fn magnitude2(&self) -> f32 {
self.dot(self)
}
fn magnitude(&self) -> f32 {
match *self {
Scalar(s) => convert(s),
Vec2(_) | Vec3(_) | Vec4(_) => self.magnitude2().sqrt(),
}
}
fn normalize(&self) -> Self {
match *self {
Scalar(_) => *self,
Vec2(_) | Vec3(_) | Vec4(_) => self.mul(1. / self.magnitude()),
}
}
}
fn mul_f32<T: RealField + SubsetOf<f32> + SupersetOf<f32>>(s: T, scalar: f32) -> T {
convert::<f32, T>(scalar) * s
}