use serde::{Deserialize, Serialize};
use std::{marker::PhantomData, rc::Rc};
mod graph_operations;
mod properties_table;
#[cfg(test)]
mod tests;
mod untyped_property;
use crate::{EasingCurve, Interpolatable, TransitionQueueEntry};
use self::properties_table::{PropertyType, PROPERTY_TIME};
use properties_table::PROPERTY_TABLE;
pub use untyped_property::UntypedProperty;
mod private {
slotmap::new_key_type!(
pub struct PropertyId;
);
}
pub trait PropertyValue: Default + Clone + Interpolatable + 'static {}
impl<T: Default + Clone + Interpolatable + 'static> PropertyValue for T {}
impl<T: PropertyValue> Interpolatable for Property<T> {
fn interpolate(&self, other: &Self, t: f64) -> Self {
let cp_self = self.clone();
let cp_other = other.clone();
Property::computed(
move || cp_self.get().interpolate(&cp_other.get(), t),
&[self.untyped(), other.untyped()],
)
}
}
#[derive(Debug, Clone)]
pub struct Property<T> {
untyped: UntypedProperty,
_phantom: PhantomData<T>,
}
impl<T: PropertyValue> Property<T> {
pub fn new(val: T) -> Self {
Self::new_optional_name(val, None)
}
pub fn computed(evaluator: impl Fn() -> T + 'static, dependents: &[UntypedProperty]) -> Self {
Self::computed_with_config(evaluator, dependents, None)
}
pub fn new_with_name(val: T, name: &str) -> Self {
Self::new_optional_name(val, Some(name))
}
pub fn computed_with_name(
evaluator: impl Fn() -> T + 'static,
dependents: &[UntypedProperty],
name: &str,
) -> Self {
Self::computed_with_config(evaluator, dependents, Some(name))
}
fn new_optional_name(val: T, name: Option<&str>) -> Self {
Self {
untyped: UntypedProperty::new(val, Vec::with_capacity(0), PropertyType::Literal, name),
_phantom: PhantomData {},
}
}
fn computed_with_config(
evaluator: impl Fn() -> T + 'static,
dependents: &[UntypedProperty],
name: Option<&str>,
) -> Self {
let inbound: Vec<_> = dependents.iter().map(|v| v.get_id()).collect();
let start_val = T::default();
let evaluator = Rc::new(evaluator);
Self {
untyped: UntypedProperty::new(
start_val,
inbound,
PropertyType::Computed { evaluator },
name,
),
_phantom: PhantomData {},
}
}
pub fn ease_to(&self, end_val: T, time: u64, curve: EasingCurve) {
self.ease_to_value(end_val, time, curve, true);
}
pub fn ease_to_later(&self, end_val: T, time: u64, curve: EasingCurve) {
self.ease_to_value(end_val, time, curve, false);
}
fn ease_to_value(&self, end_val: T, time: u64, curve: EasingCurve, overwrite: bool) {
PROPERTY_TABLE.with(|t| {
t.transition(
self.untyped.id,
TransitionQueueEntry {
duration_frames: time,
curve,
ending_value: end_val,
},
overwrite,
)
})
}
pub fn get(&self) -> T {
PROPERTY_TABLE.with(|t| t.get_value(self.untyped.id))
}
pub fn set(&self, val: T) {
PROPERTY_TABLE.with(|t| t.set_value(self.untyped.id, val));
}
pub fn update(&self, f: impl FnOnce(&mut T)) {
let mut val = self.get();
f(&mut val);
self.set(val);
}
pub fn read<V>(&self, f: impl FnOnce(&T) -> V) -> V {
let val = self.get();
f(&val)
}
pub fn replace_with(&self, target: Property<T>) {
PROPERTY_TABLE.with(|t| {
t.replace_property_keep_outbound_connections::<T>(self.untyped.id, target.untyped.id)
})
}
pub fn untyped(&self) -> UntypedProperty {
self.untyped.clone()
}
}
impl<T: PropertyValue> Default for Property<T> {
fn default() -> Self {
Property::new(T::default())
}
}
impl<'de, T: PropertyValue + Deserialize<'de>> Deserialize<'de> for Property<T> {
fn deserialize<D>(deserializer: D) -> Result<Property<T>, D::Error>
where
D: serde::Deserializer<'de>,
{
let value = T::deserialize(deserializer)?;
Ok(Property::new(value))
}
}
impl<T: PropertyValue + Serialize> Serialize for Property<T> {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: serde::Serializer,
{
self.get().serialize(serializer)
}
}
pub fn property_table_total_properties_count() -> usize {
PROPERTY_TABLE.with(|t| t.total_properties_count())
}
pub fn register_time(prop: &Property<u64>) {
PROPERTY_TIME.with_borrow_mut(|time| *time = prop.clone());
}