agility 0.1.1

A powerful and elegant reactive programming library for Rust, inspired by category theory
Documentation
use crate::signal::Signal;
use crate::signal_sync::SignalSync;

trait Mutable {}

/// Trait for lifting collections of signals into a single signal.
/// For example, lifting (Signal<A>, Signal<B>) into Signal<(A, B)>
pub trait LiftInto<T> {
    /// Lift the collection into a single signal
    ///
    /// # Examples
    /// ```rust
    /// let a = Signal::new(1);
    /// let b = Signal::new(2);
    /// let lifted = [a, b].lift();
    /// lifted.with(|[x, y]| println!("Lifted: {}, {}", x, y));
    /// a.send(10); // prints "Lifted: 10, 2"
    /// b.send(20); // prints "Lifted: 10, 20"
    /// ```
    fn lift(self) -> T;
}

/// Trait for lifting collections of thread-safe signals into a single thread-safe signal.
/// For example, lifting (SignalSync<A>, SignalSync<B>) into SignalSync<(A, B)>
pub trait LiftIntoSync<T> {
    /// Lift the collection into a single thread-safe signal
    fn lift(self) -> T;
}

/// Trait for signal-like types that can be lifted
pub trait Liftable<'a> {
    type Inner;
    /// Get a reference to the underlying signal
    fn as_ref(&self) -> &Signal<'a, Self::Inner>;
}

/// Trait for thread-safe signal-like types that can be lifted
pub trait LiftableSync<'a> {
    type Inner;
    /// Get a reference to the underlying thread-safe signal
    fn as_ref(&self) -> &SignalSync<'a, Self::Inner>;
}

impl<T> Mutable for Vec<T> {}

impl<'a, T> Liftable<'a> for Signal<'a, T> {
    type Inner = T;
    fn as_ref(&self) -> &Signal<'a, Self::Inner> {
        &self
    }
}

impl<'a, T> Liftable<'a> for &Signal<'a, T> {
    type Inner = T;
    fn as_ref(&self) -> &Signal<'a, Self::Inner> {
        self
    }
}

impl<'a, T: Send + Sync + 'a> LiftableSync<'a> for SignalSync<'a, T> {
    type Inner = T;
    fn as_ref(&self) -> &SignalSync<'a, Self::Inner> {
        &self
    }
}

impl<'a, T: Send + Sync + 'a> LiftableSync<'a> for &SignalSync<'a, T> {
    type Inner = T;
    fn as_ref(&self) -> &SignalSync<'a, Self::Inner> {
        self
    }
}

impl<'a, S1, S2> LiftInto<Signal<'a, (S1::Inner, S2::Inner)>> for (S1, S2)
where
    S1: Liftable<'a>,
    S2: Liftable<'a>,
    S1::Inner: 'a,
    S2::Inner: 'a,
{
    fn lift(self) -> Signal<'a, (S1::Inner, S2::Inner)> {
        self.0.as_ref().combine(self.1.as_ref())
    }
}

impl<'a, S, I> LiftInto<Signal<'a, Vec<S::Inner>>> for I
where
    S: Liftable<'a>,
    I: IntoIterator<Item = S> + Mutable,
    S::Inner: 'a,
{
    fn lift(self) -> Signal<'a, Vec<S::Inner>> {
        let mut items: Vec<S> = self.into_iter().collect();
        if items.is_empty() {
            Signal::new(Vec::new())
        } else {
            let first = items.remove(0);
            first.as_ref().extend(items.into_iter())
        }
    }
}

impl<'a, const N: usize, S> LiftInto<Signal<'a, [S::Inner; N]>> for [S; N]
where
    S: Liftable<'a>,
    S::Inner: 'a,
{
    fn lift(self) -> Signal<'a, [S::Inner; N]> {
        Signal::<S::Inner>::lift_from_array::<S, N>(self)
    }
}

// LiftIntoSync implementations for thread-safe variant
impl<'a, S1, S2> LiftIntoSync<SignalSync<'a, (S1::Inner, S2::Inner)>> for (S1, S2)
where
    S1: LiftableSync<'a>,
    S2: LiftableSync<'a>,
    S1::Inner: Send + Sync + 'a,
    S2::Inner: Send + Sync + 'a,
{
    fn lift(self) -> SignalSync<'a, (S1::Inner, S2::Inner)> {
        self.0.as_ref().combine(self.1.as_ref())
    }
}

impl<'a, S, I> LiftIntoSync<SignalSync<'a, Vec<S::Inner>>> for I
where
    S: LiftableSync<'a>,
    I: IntoIterator<Item = S> + Mutable,
    S::Inner: Send + Sync + 'a,
{
    fn lift(self) -> SignalSync<'a, Vec<S::Inner>> {
        let mut items: Vec<S> = self.into_iter().collect();
        if items.is_empty() {
            SignalSync::new(Vec::new())
        } else {
            let first = items.remove(0);
            first.as_ref().extend(items.into_iter())
        }
    }
}

impl<'a, const N: usize, S> LiftIntoSync<SignalSync<'a, [S::Inner; N]>> for [S; N]
where
    S: LiftableSync<'a>,
    S::Inner: Send + Sync + 'a,
{
    fn lift(self) -> SignalSync<'a, [S::Inner; N]> {
        SignalSync::<S::Inner>::lift_from_array::<S, N>(self)
    }
}