use crate::props::{Props, PropsData, PropsError};
use std::{any::TypeId, sync::mpsc::Sender};
#[derive(Debug, Clone)]
pub enum StateError {
Props(PropsError),
CouldNotWriteChange,
}
#[derive(Debug, Clone)]
pub enum StateChange {
Set(Props),
Include(Props),
Exclude(TypeId),
}
#[derive(Clone)]
pub struct StateUpdate(Sender<StateChange>);
impl StateUpdate {
pub fn new(sender: Sender<StateChange>) -> Self {
Self(sender)
}
pub fn set<T>(&self, data: T) -> Result<(), StateError>
where
T: Into<Props>,
{
if self.0.send(StateChange::Set(data.into())).is_err() {
Err(StateError::CouldNotWriteChange)
} else {
Ok(())
}
}
pub fn include<T>(&self, data: T) -> Result<(), StateError>
where
T: Into<Props>,
{
let data = data.into();
if self.0.send(StateChange::Include(data)).is_err() {
Err(StateError::CouldNotWriteChange)
} else {
Ok(())
}
}
pub fn exclude<T>(&self) -> Result<(), StateError>
where
T: 'static + PropsData,
{
if self
.0
.send(StateChange::Exclude(TypeId::of::<T>()))
.is_err()
{
Err(StateError::CouldNotWriteChange)
} else {
Ok(())
}
}
}
pub struct State<'a> {
data: &'a Props,
update: StateUpdate,
}
impl<'a> State<'a> {
pub fn new(data: &'a Props, update: StateUpdate) -> Self {
Self { data, update }
}
#[inline]
pub fn data(&self) -> &Props {
self.data
}
pub fn has<T>(&self) -> bool
where
T: 'static + PropsData,
{
self.data.has::<T>()
}
pub fn read<T>(&self) -> Result<&'a T, StateError>
where
T: 'static + PropsData,
{
match self.data.read() {
Ok(v) => Ok(v),
Err(e) => Err(StateError::Props(e)),
}
}
pub fn map_or_default<T, R, F>(&self, f: F) -> R
where
T: 'static + PropsData,
R: Default,
F: FnMut(&T) -> R,
{
self.data.map_or_default(f)
}
pub fn map_or_else<T, R, F, E>(&self, f: F, e: E) -> R
where
T: 'static + PropsData,
F: FnMut(&T) -> R,
E: FnMut() -> R,
{
self.data.map_or_else(f, e)
}
pub fn read_cloned<T>(&self) -> Result<T, StateError>
where
T: 'static + PropsData + Clone,
{
match self.data.read_cloned() {
Ok(v) => Ok(v),
Err(e) => Err(StateError::Props(e)),
}
}
pub fn read_cloned_or_default<T>(&self) -> T
where
T: 'static + PropsData + Clone + Default,
{
self.data.read_cloned_or_default()
}
pub fn read_cloned_or_else<T, F>(&self, f: F) -> T
where
T: 'static + PropsData + Clone + Default,
F: FnMut() -> T,
{
self.data.read_cloned_or_else(f)
}
pub fn write<T>(&self, data: T) -> Result<(), StateError>
where
T: 'static + PropsData + Send + Sync,
{
self.update.set(data)
}
pub fn write_with<T>(&self, data: T) -> Result<(), StateError>
where
T: 'static + PropsData + Send + Sync,
{
self.update.include(data)
}
pub fn write_without<T>(&self) -> Result<(), StateError>
where
T: 'static + PropsData + Send + Sync,
{
self.update.exclude::<T>()
}
pub fn mutate<T, F>(&self, mut f: F) -> Result<(), StateError>
where
T: 'static + PropsData + Send + Sync,
F: FnMut(&T) -> T,
{
match self.read() {
Ok(data) => {
let data = f(data);
self.write(data)
}
Err(error) => Err(error),
}
}
pub fn mutate_cloned<T, F>(&self, mut f: F) -> Result<(), StateError>
where
T: 'static + PropsData + Send + Sync + Clone,
F: FnMut(&mut T),
{
match self.read::<T>() {
Ok(data) => {
let mut data = data.clone();
f(&mut data);
self.write(data)
}
Err(error) => Err(error),
}
}
pub fn update(&self) -> &StateUpdate {
&self.update
}
}