use crate::{traits::DITarget, types::DI};
use std::{any::Any, collections::HashMap, future::Future};
#[cfg(target_arch = "wasm32")]
use std::cell::RefCell;
#[cfg(not(target_arch = "wasm32"))]
use std::sync::RwLock;
#[derive(Debug)]
pub struct DIContainer {
#[cfg(target_arch = "wasm32")]
components: RefCell<HashMap<String, DI<dyn Any>>>,
#[cfg(not(target_arch = "wasm32"))]
components: RwLock<HashMap<String, DI<dyn Any + Send + Sync>>>,
}
impl DIContainer {
pub fn new() -> DIContainer {
DIContainer {
#[cfg(target_arch = "wasm32")]
components: RefCell::new(HashMap::new()),
#[cfg(not(target_arch = "wasm32"))]
components: RwLock::new(HashMap::new()),
}
}
pub fn get<T: DITarget>(&self) -> Option<DI<T>> {
#[cfg(target_arch = "wasm32")]
let comps = self.components.borrow();
#[cfg(not(target_arch = "wasm32"))]
let comps = self.components.read().unwrap();
comps
.get(std::any::type_name::<T>())
.map(|c| c.clone().downcast::<T>().unwrap())
}
pub fn put_if_absent<T: DITarget>(&self, c: &DI<T>) -> DI<T> {
#[cfg(target_arch = "wasm32")]
let mut components = self.components.borrow_mut();
#[cfg(not(target_arch = "wasm32"))]
let mut components = self.components.write().unwrap();
let key = std::any::type_name::<T>();
let value = components
.get(key)
.map(|c| c.clone().downcast::<T>().unwrap());
if let Some(c) = value {
c
} else {
components.insert(key.into(), c.clone());
c.clone()
}
}
pub fn get_or_init<T, F>(&self, init: F) -> DI<T>
where
T: DITarget,
F: Fn() -> T,
{
if let Some(c) = self.get::<T>() {
c
} else {
let c = DI::new(init());
self.put_if_absent(&c)
}
}
pub async fn get_or_init_async<T, F, Fut>(&self, init: F) -> DI<T>
where
T: DITarget,
F: Fn() -> Fut,
Fut: Future<Output = T>,
{
if let Some(c) = self.get::<T>() {
c
} else {
let v = init().await;
let c = DI::new(v);
self.put_if_absent(&c)
}
}
}