use std::sync::Arc;
use dioxus_core::{
    prelude::spawn,
    use_hook,
    AttributeValue,
};
use dioxus_signals::{
    ReadOnlySignal,
    Readable,
    Signal,
    Writable,
};
use freya_common::NodeReferenceLayout;
use freya_node_state::{
    CustomAttributeValues,
    NodeReference,
};
use tokio::sync::watch::channel;
pub fn use_node() -> (AttributeValue, NodeReferenceLayout) {
    let (tx, signal) = use_hook(|| {
        let (tx, mut rx) = channel::<NodeReferenceLayout>(NodeReferenceLayout::default());
        let mut signal = Signal::new(NodeReferenceLayout::default());
        spawn(async move {
            while rx.changed().await.is_ok() {
                if *signal.peek() != *rx.borrow() {
                    signal.set(rx.borrow().clone());
                }
            }
        });
        (Arc::new(tx), signal)
    });
    (
        AttributeValue::any_value(CustomAttributeValues::Reference(NodeReference(tx))),
        signal.read_unchecked().clone(),
    )
}
pub fn use_node_signal() -> (AttributeValue, ReadOnlySignal<NodeReferenceLayout>) {
    let (tx, signal) = use_hook(|| {
        let (tx, mut rx) = channel::<NodeReferenceLayout>(NodeReferenceLayout::default());
        let mut signal = Signal::new(NodeReferenceLayout::default());
        spawn(async move {
            while rx.changed().await.is_ok() {
                if *signal.peek() != *rx.borrow() {
                    signal.set(rx.borrow().clone());
                }
            }
        });
        (Arc::new(tx), signal)
    });
    (
        AttributeValue::any_value(CustomAttributeValues::Reference(NodeReference(tx))),
        signal.into(),
    )
}
pub fn use_node_signal_with_prev() -> (
    AttributeValue,
    ReadOnlySignal<Option<NodeReferenceLayout>>,
    ReadOnlySignal<Option<NodeReferenceLayout>>,
) {
    let (tx, curr_signal, prev_signal) = use_hook(|| {
        let (tx, mut rx) = channel::<NodeReferenceLayout>(NodeReferenceLayout::default());
        let mut curr_signal = Signal::new(None);
        let mut prev_signal = Signal::new(None);
        spawn(async move {
            while rx.changed().await.is_ok() {
                if *curr_signal.peek() != Some(rx.borrow().clone()) {
                    prev_signal.set(curr_signal());
                    curr_signal.set(Some(rx.borrow().clone()));
                }
            }
        });
        (Arc::new(tx), curr_signal, prev_signal)
    });
    (
        AttributeValue::any_value(CustomAttributeValues::Reference(NodeReference(tx))),
        curr_signal.into(),
        prev_signal.into(),
    )
}
#[cfg(test)]
mod test {
    use freya::prelude::*;
    use freya_testing::prelude::*;
    use crate::use_node;
    #[tokio::test]
    pub async fn track_size() {
        fn use_node_app() -> Element {
            let (reference, size) = use_node();
            rsx!(
                rect {
                    reference: reference,
                    width: "50%",
                    height: "25%",
                    label {
                        "{size.area.width()}"
                    }
                }
            )
        }
        let mut utils = launch_test_with_config(
            use_node_app,
            TestingConfig::<()> {
                size: (500.0, 800.0).into(),
                ..TestingConfig::default()
            },
        );
        utils.wait_for_update().await;
        let root = utils.root().get(0);
        assert_eq!(
            root.get(0).get(0).text().unwrap().parse::<f32>(),
            Ok(500.0 * 0.5)
        );
        utils.resize((300.0, 800.0).into());
        utils.wait_for_update().await;
        let root = utils.root().get(0);
        assert_eq!(
            root.get(0).get(0).text().unwrap().parse::<f32>(),
            Ok(300.0 * 0.5)
        );
    }
}