use std::cell::RefCell;
use std::rc::Rc;
use std::sync::atomic::{AtomicUsize, Ordering};
use crate::reactive;
pub type SubId = usize;
static NEXT_SIGNAL_ID: AtomicUsize = AtomicUsize::new(1);
pub struct Signal<T: 'static>(Rc<RefCell<Inner<T>>>);
impl<T> Clone for Signal<T> {
fn clone(&self) -> Self {
Self(self.0.clone())
}
}
struct Inner<T> {
id: usize,
value: T,
subs: Vec<Option<Box<dyn Fn(&T)>>>,
}
impl<T> Signal<T> {
pub fn new(value: T) -> Self {
let id = NEXT_SIGNAL_ID.fetch_add(1, Ordering::Relaxed);
Self(Rc::new(RefCell::new(Inner {
id,
value,
subs: Vec::new(),
})))
}
pub fn id(&self) -> usize {
self.0.borrow().id
}
pub fn get(&self) -> T
where
T: Clone,
{
let inner = self.0.borrow();
reactive::register_signal_read(inner.id);
inner.value.clone()
}
pub fn set(&self, v: T) {
let id = {
let mut inner = self.0.borrow_mut();
inner.value = v;
inner.id
};
{
let inner = self.0.borrow();
let vref = &inner.value;
for s in &inner.subs {
if let Some(cb) = s.as_ref() {
cb(vref);
}
}
}
reactive::signal_changed(id);
crate::request_frame();
}
pub fn update<F: FnOnce(&mut T)>(&self, f: F) {
let id = {
let mut inner = self.0.borrow_mut();
f(&mut inner.value);
inner.id
};
{
let inner = self.0.borrow();
let vref = &inner.value;
for s in &inner.subs {
if let Some(cb) = s.as_ref() {
cb(vref);
}
}
}
reactive::signal_changed(id);
crate::request_frame();
}
pub fn subscribe(&self, f: impl Fn(&T) + 'static) -> SubId {
self.0.borrow_mut().subs.push(Some(Box::new(f)));
self.0.borrow().subs.len() - 1
}
pub fn unsubscribe(&self, id: SubId) -> bool {
let mut inner = self.0.borrow_mut();
if id < inner.subs.len() {
inner.subs[id] = None;
true
} else {
false
}
}
pub fn subscribe_guard(&self, f: impl Fn(&T) + 'static) -> SubGuard<T> {
let id = self.subscribe(f);
let sig = self.clone();
SubGuard {
sig,
id,
active: true,
}
}
}
pub fn signal<T>(t: T) -> Signal<T> {
Signal::new(t)
}
pub struct SubGuard<T: 'static> {
sig: crate::Signal<T>,
id: SubId,
active: bool,
}
impl<T> Drop for SubGuard<T> {
fn drop(&mut self) {
if self.active {
let _ = self.sig.unsubscribe(self.id);
self.active = false;
}
}
}