#![allow(unused)] use dioxus::prelude::*; fn main() {} struct AppSettings {} // ANCHOR: wrap_context fn use_settings(cx: &ScopeState) -> &UseSharedState { use_shared_state::(cx).expect("App settings not provided") } // ANCHOR_END: wrap_context // ANCHOR: use_storage use gloo_storage::{LocalStorage, Storage}; use serde::{de::DeserializeOwned, Serialize}; /// A persistent storage hook that can be used to store data across application reloads. #[allow(clippy::needless_return)] pub fn use_persistent( cx: &ScopeState, // A unique key for the storage entry key: impl ToString, // A function that returns the initial value if the storage entry is empty init: impl FnOnce() -> T, ) -> &UsePersistent { // Use the use_ref hook to create a mutable state for the storage entry let state = use_ref(cx, move || { // This closure will run when the hook is created let key = key.to_string(); let value = LocalStorage::get(key.as_str()).ok().unwrap_or_else(init); StorageEntry { key, value } }); // Wrap the state in a new struct with a custom API // Note: We use use_hook here so that this hook is easier to use in closures in the rsx. Any values with the same lifetime as the ScopeState can be used in the closure without cloning. cx.use_hook(|| UsePersistent { inner: state.clone(), }) } struct StorageEntry { key: String, value: T, } /// Storage that persists across application reloads pub struct UsePersistent { inner: UseRef>, } impl UsePersistent { /// Returns a reference to the value pub fn get(&self) -> T { self.inner.read().value.clone() } /// Sets the value pub fn set(&self, value: T) { let mut inner = self.inner.write(); // Write the new value to local storage LocalStorage::set(inner.key.as_str(), &value); inner.value = value; } } // ANCHOR_END: use_storage