use dioxus_core::prelude::Context; use std::{ cell::{Cell, Ref, RefCell, RefMut}, fmt::Display, ops::{Deref, DerefMut, Not}, rc::Rc, }; /// Store state between component renders! /// /// ## The "Pinnacle" of state hooks /// /// The Dioxus version of `useState` is the "king daddy" of state management. It allows you to ergonomically store and /// modify state between component renders. When the state is updated, the component will re-render. /// /// Dioxus' use_state basically wraps a RefCell with helper methods and integrates it with the VirtualDOM update system. /// /// [`use_state`] exposes a few helper methods to modify the underlying state: /// - `.set(new)` allows you to override the "work in progress" value with a new value /// - `.get_mut()` allows you to modify the WIP value /// - `.get_wip()` allows you to access the WIP value /// - `.deref()` provides the previous value (often done implicitly, though a manual dereference with `*` might be required) /// /// Additionally, a ton of std::ops traits are implemented for the `UseState` wrapper, meaning any mutative type operations /// will automatically be called on the WIP value. /// /// ## Combinators /// /// On top of the methods to set/get state, `use_state` also supports fancy combinators to extend its functionality: /// - `.classic()` and `.split()` convert the hook into the classic React-style hook /// ```rust /// let (state, set_state) = use_state(cx, || 10).split() /// ``` /// /// /// Usage: /// ```ignore /// const Example: FC<()> = |cx, props|{ /// let counter = use_state(cx, || 0); /// let increment = |_| counter += 1; /// let decrement = |_| counter += 1; /// /// html! { ///
///

"Counter: {counter}"

/// /// ///
/// } /// } /// ``` pub fn use_state<'a, T: 'static>( cx: Context<'a>, initial_state_fn: impl FnOnce() -> T, ) -> UseState<'a, T> { cx.use_hook( move |_| UseStateInner { current_val: initial_state_fn(), update_callback: cx.schedule_update(), wip: Rc::new(RefCell::new(None)), update_scheuled: Cell::new(false), }, move |hook| { hook.update_scheuled.set(false); let mut new_val = hook.wip.borrow_mut(); if new_val.is_some() { hook.current_val = new_val.take().unwrap(); } UseState { inner: &*hook } }, |_| {}, ) } struct UseStateInner { current_val: T, update_scheuled: Cell, update_callback: Rc, wip: Rc>>, } pub struct UseState<'a, T: 'static> { inner: &'a UseStateInner, } impl Copy for UseState<'_, T> {} impl<'a, T> Clone for UseState<'a, T> where T: 'static, { fn clone(&self) -> Self { UseState { inner: self.inner } } } impl<'a, T: 'static> UseState<'a, T> { /// Tell the Dioxus Scheduler that we need to be processed pub fn needs_update(&self) { if !self.inner.update_scheuled.get() { self.inner.update_scheuled.set(true); (self.inner.update_callback)(); } } pub fn set(&self, new_val: T) { *self.inner.wip.borrow_mut() = Some(new_val); self.needs_update(); } pub fn get(&self) -> &'a T { &self.inner.current_val } /// Get the current status of the work-in-progress data pub fn get_wip(&self) -> Ref> { self.inner.wip.borrow() } /// Get the current status of the work-in-progress data pub fn get_wip_mut(&self) -> RefMut> { self.inner.wip.borrow_mut() } pub fn classic(self) -> (&'a T, &'a Rc) { todo!() } pub fn setter(&self) -> Rc { let slot = self.inner.wip.clone(); Rc::new(move |new| *slot.borrow_mut() = Some(new)) } pub fn for_async(&self) -> AsyncUseState { AsyncUseState { wip: self.inner.wip.clone(), } } pub fn split_for_async(&'a self) -> (&'a Self, AsyncUseState) { (self, self.for_async()) } } impl<'a, T: 'static + ToOwned> UseState<'a, T> { /// Gain mutable access to the new value. This method is only available when the value is a `ToOwned` type. /// /// Mutable access is derived by calling "ToOwned" (IE cloning) on the current value. /// /// To get a reference to the current value, use `.get()` pub fn modify(self) -> RefMut<'a, T> { // make sure we get processed self.needs_update(); // Bring out the new value, cloning if it we need to // "get_mut" is locked behind "ToOwned" to make it explicit that cloning occurs to use this RefMut::map(self.inner.wip.borrow_mut(), |slot| { if slot.is_none() { *slot = Some(self.inner.current_val.to_owned()); } slot.as_mut().unwrap() }) } } impl<'a, T> std::ops::Deref for UseState<'a, T> { type Target = T; fn deref(&self) -> &Self::Target { self.get() } } use std::ops::{Add, AddAssign, Div, DivAssign, Mul, MulAssign, Sub, SubAssign}; impl<'a, T: Copy + Add> Add for UseState<'a, T> { type Output = T; fn add(self, rhs: T) -> Self::Output { self.inner.current_val.add(rhs) } } impl<'a, T: Copy + Add> AddAssign for UseState<'a, T> { fn add_assign(&mut self, rhs: T) { self.set(self.inner.current_val.add(rhs)); } } impl<'a, T: Copy + Sub> Sub for UseState<'a, T> { type Output = T; fn sub(self, rhs: T) -> Self::Output { self.inner.current_val.sub(rhs) } } impl<'a, T: Copy + Sub> SubAssign for UseState<'a, T> { fn sub_assign(&mut self, rhs: T) { self.set(self.inner.current_val.sub(rhs)); } } /// MUL impl<'a, T: Copy + Mul> Mul for UseState<'a, T> { type Output = T; fn mul(self, rhs: T) -> Self::Output { self.inner.current_val.mul(rhs) } } impl<'a, T: Copy + Mul> MulAssign for UseState<'a, T> { fn mul_assign(&mut self, rhs: T) { self.set(self.inner.current_val.mul(rhs)); } } /// DIV impl<'a, T: Copy + Div> Div for UseState<'a, T> { type Output = T; fn div(self, rhs: T) -> Self::Output { self.inner.current_val.div(rhs) } } impl<'a, T: Copy + Div> DivAssign for UseState<'a, T> { fn div_assign(&mut self, rhs: T) { self.set(self.inner.current_val.div(rhs)); } } impl<'a, V, T: PartialEq> PartialEq for UseState<'a, T> { fn eq(&self, other: &V) -> bool { self.get() == other } } impl<'a, O, T: Not + Copy> Not for UseState<'a, T> { type Output = O; fn not(self) -> Self::Output { !*self.get() } } // enable displaty for the handle impl<'a, T: 'static + Display> std::fmt::Display for UseState<'a, T> { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { write!(f, "{}", self.inner.current_val) } } /// A less ergonmic but still capable form of use_state that's valid for `static lifetime pub struct AsyncUseState { wip: Rc>>, } impl AsyncUseState { pub fn get_mut<'a>(&'a self) -> RefMut<'a, T> { // make sure we get processed // self.needs_update(); // Bring out the new value, cloning if it we need to // "get_mut" is locked behind "ToOwned" to make it explicit that cloning occurs to use this RefMut::map(self.wip.borrow_mut(), |slot| { // slot.as_mut().unwrap() }) } pub fn set(&mut self, val: T) { *self.wip.borrow_mut() = Some(val); } }