123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212 |
- use std::ops::DerefMut;
- use crate::read::Readable;
- /// A trait for states that can be read from like [`crate::Signal`], or [`crate::GlobalSignal`]. You may choose to accept this trait as a parameter instead of the concrete type to allow for more flexibility in your API. For example, instead of creating two functions, one that accepts a [`crate::Signal`] and one that accepts a [`crate::GlobalSignal`], you can create one function that accepts a [`Writable`] type.
- pub trait Writable<T: 'static>: Readable<T> {
- /// The type of the reference.
- type Mut<R: ?Sized + 'static>: DerefMut<Target = R>;
- /// Map the reference to a new type.
- fn map_mut<I, U: ?Sized, F: FnOnce(&mut I) -> &mut U>(ref_: Self::Mut<I>, f: F)
- -> Self::Mut<U>;
- /// Try to map the reference to a new type.
- fn try_map_mut<I, U: ?Sized, F: FnOnce(&mut I) -> Option<&mut U>>(
- ref_: Self::Mut<I>,
- f: F,
- ) -> Option<Self::Mut<U>>;
- /// Get a mutable reference to the value. If the value has been dropped, this will panic.
- #[track_caller]
- fn write(&mut self) -> Self::Mut<T> {
- self.try_write().unwrap()
- }
- /// Try to get a mutable reference to the value. If the value has been dropped, this will panic.
- fn try_write(&self) -> Result<Self::Mut<T>, generational_box::BorrowMutError>;
- /// Run a function with a mutable reference to the value. If the value has been dropped, this will panic.
- #[track_caller]
- fn with_mut<O>(&mut self, f: impl FnOnce(&mut T) -> O) -> O {
- f(&mut *self.write())
- }
- /// Set the value of the signal. This will trigger an update on all subscribers.
- #[track_caller]
- fn set(&mut self, value: T) {
- *self.write() = value;
- }
- /// Invert the boolean value of the signal. This will trigger an update on all subscribers.
- #[track_caller]
- fn toggle(&mut self)
- where
- T: std::ops::Not<Output = T> + Clone,
- {
- self.set(!self.cloned());
- }
- /// Index into the inner value and return a reference to the result.
- #[track_caller]
- fn index_mut<I>(&mut self, index: I) -> Self::Mut<T::Output>
- where
- T: std::ops::IndexMut<I>,
- {
- Self::map_mut(self.write(), |v| v.index_mut(index))
- }
- /// Takes the value out of the Signal, leaving a Default in its place.
- #[track_caller]
- fn take(&mut self) -> T
- where
- T: Default,
- {
- self.with_mut(|v| std::mem::take(v))
- }
- /// Replace the value in the Signal, returning the old value.
- #[track_caller]
- fn replace(&mut self, value: T) -> T {
- self.with_mut(|v| std::mem::replace(v, value))
- }
- }
- /// An extension trait for Writable<Option<T>> that provides some convenience methods.
- pub trait WritableOptionExt<T: 'static>: Writable<Option<T>> {
- /// Gets the value out of the Option, or inserts the given value if the Option is empty.
- fn get_or_insert(&mut self, default: T) -> Self::Mut<T> {
- self.get_or_insert_with(|| default)
- }
- /// Gets the value out of the Option, or inserts the value returned by the given function if the Option is empty.
- fn get_or_insert_with(&mut self, default: impl FnOnce() -> T) -> Self::Mut<T> {
- let borrow = self.read();
- if borrow.is_none() {
- drop(borrow);
- self.with_mut(|v| *v = Some(default()));
- Self::map_mut(self.write(), |v| v.as_mut().unwrap())
- } else {
- Self::map_mut(self.write(), |v| v.as_mut().unwrap())
- }
- }
- /// Attempts to write the inner value of the Option.
- #[track_caller]
- fn as_mut(&mut self) -> Option<Self::Mut<T>> {
- Self::try_map_mut(self.write(), |v: &mut Option<T>| v.as_mut())
- }
- }
- impl<T, W> WritableOptionExt<T> for W
- where
- T: 'static,
- W: Writable<Option<T>>,
- {
- }
- /// An extension trait for Writable<Vec<T>> that provides some convenience methods.
- pub trait WritableVecExt<T: 'static>: Writable<Vec<T>> {
- /// Pushes a new value to the end of the vector.
- #[track_caller]
- fn push(&mut self, value: T) {
- self.with_mut(|v| v.push(value))
- }
- /// Pops the last value from the vector.
- #[track_caller]
- fn pop(&mut self) -> Option<T> {
- self.with_mut(|v| v.pop())
- }
- /// Inserts a new value at the given index.
- #[track_caller]
- fn insert(&mut self, index: usize, value: T) {
- self.with_mut(|v| v.insert(index, value))
- }
- /// Removes the value at the given index.
- #[track_caller]
- fn remove(&mut self, index: usize) -> T {
- self.with_mut(|v| v.remove(index))
- }
- /// Clears the vector, removing all values.
- #[track_caller]
- fn clear(&mut self) {
- self.with_mut(|v| v.clear())
- }
- /// Extends the vector with the given iterator.
- #[track_caller]
- fn extend(&mut self, iter: impl IntoIterator<Item = T>) {
- self.with_mut(|v| v.extend(iter))
- }
- /// Truncates the vector to the given length.
- #[track_caller]
- fn truncate(&mut self, len: usize) {
- self.with_mut(|v| v.truncate(len))
- }
- /// Swaps two values in the vector.
- #[track_caller]
- fn swap_remove(&mut self, index: usize) -> T {
- self.with_mut(|v| v.swap_remove(index))
- }
- /// Retains only the values that match the given predicate.
- #[track_caller]
- fn retain(&mut self, f: impl FnMut(&T) -> bool) {
- self.with_mut(|v| v.retain(f))
- }
- /// Splits the vector into two at the given index.
- #[track_caller]
- fn split_off(&mut self, at: usize) -> Vec<T> {
- self.with_mut(|v| v.split_off(at))
- }
- /// Try to mutably get an element from the vector.
- #[track_caller]
- fn get_mut(&mut self, index: usize) -> Option<Self::Mut<T>> {
- Self::try_map_mut(self.write(), |v: &mut Vec<T>| v.get_mut(index))
- }
- /// Gets an iterator over the values of the vector.
- #[track_caller]
- fn iter_mut(&self) -> WritableValueIterator<T, Self>
- where
- Self: Sized + Clone,
- {
- WritableValueIterator {
- index: 0,
- value: self.clone(),
- phantom: std::marker::PhantomData,
- }
- }
- }
- /// An iterator over the values of a `Writable<Vec<T>>`.
- pub struct WritableValueIterator<T, R> {
- index: usize,
- value: R,
- phantom: std::marker::PhantomData<T>,
- }
- impl<T: 'static, R: Writable<Vec<T>>> Iterator for WritableValueIterator<T, R> {
- type Item = R::Mut<T>;
- fn next(&mut self) -> Option<Self::Item> {
- let index = self.index;
- self.index += 1;
- self.value.get_mut(index)
- }
- }
- impl<T, W> WritableVecExt<T> for W
- where
- T: 'static,
- W: Writable<Vec<T>>,
- {
- }
|