use std::sync::Arc;
use parking_lot::RwLock;
use serde::{Deserialize, Serialize};
use serde_json::Value;
use super::context::current_context;
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash, Serialize, Deserialize)]
pub struct SignalId(pub u32);
impl std::fmt::Display for SignalId {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "s{}", self.0)
}
}
struct SignalInner<T> {
id: SignalId,
value: RwLock<T>,
subscribers: RwLock<Vec<u32>>,
}
pub struct Signal<T> {
inner: Arc<SignalInner<T>>,
}
impl<T> Clone for Signal<T> {
fn clone(&self) -> Self {
Self {
inner: self.inner.clone(),
}
}
}
impl<T> Signal<T>
where
T: Clone + Serialize + 'static,
{
pub fn new(initial: T) -> Self {
let id = current_context()
.map(|ctx| ctx.next_signal_id())
.unwrap_or_else(fallback_id);
let signal = Self {
inner: Arc::new(SignalInner {
id,
value: RwLock::new(initial),
subscribers: RwLock::new(Vec::new()),
}),
};
if let Some(ctx) = current_context() {
ctx.register_signal(id, signal.serialize_value());
}
signal
}
pub fn id(&self) -> SignalId {
self.inner.id
}
pub fn peek(&self) -> T {
self.inner.value.read().clone()
}
pub fn get(&self) -> T {
self.track();
self.peek()
}
pub fn set(&self, value: T) {
*self.inner.value.write() = value;
self.notify();
}
pub fn update<F>(&self, f: F)
where
F: FnOnce(&mut T),
{
let mut guard = self.inner.value.write();
f(&mut guard);
drop(guard);
self.notify();
}
fn track(&self) {
if let Some(ctx) = current_context() {
if let Some(eid) = ctx.current_effect_id() {
let mut subs = self.inner.subscribers.write();
if !subs.contains(&eid) {
subs.push(eid);
}
}
}
}
fn notify(&self) {
if let Some(ctx) = current_context() {
let subs = self.inner.subscribers.read().clone();
for eid in subs {
ctx.run_effect(eid);
}
ctx.update_signal(self.inner.id, self.serialize_value());
}
}
fn serialize_value(&self) -> Value {
serde_json::to_value(&*self.inner.value.read()).unwrap_or(Value::Null)
}
pub fn split(self) -> (ReadSignal<T>, WriteSignal<T>) {
(
ReadSignal {
signal: self.clone(),
},
WriteSignal { signal: self },
)
}
}
#[derive(Clone)]
pub struct ReadSignal<T> {
signal: Signal<T>,
}
impl<T: Clone + Serialize + 'static> ReadSignal<T> {
pub fn id(&self) -> SignalId {
self.signal.id()
}
pub fn get(&self) -> T {
self.signal.get()
}
pub fn peek(&self) -> T {
self.signal.peek()
}
}
#[derive(Clone)]
pub struct WriteSignal<T> {
signal: Signal<T>,
}
impl<T: Clone + Serialize + 'static> WriteSignal<T> {
pub fn id(&self) -> SignalId {
self.signal.id()
}
pub fn set(&self, value: T) {
self.signal.set(value)
}
pub fn update<F>(&self, f: F)
where
F: FnOnce(&mut T),
{
self.signal.update(f)
}
}
pub fn use_signal<T: Clone + Serialize + 'static>(initial: T) -> Signal<T> {
Signal::new(initial)
}
fn fallback_id() -> SignalId {
use std::sync::atomic::{AtomicU32, Ordering};
static COUNTER: AtomicU32 = AtomicU32::new(1_000_000);
SignalId(COUNTER.fetch_add(1, Ordering::Relaxed))
}