1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
use std::{
    cell::{Ref, RefCell},
    rc::Rc,
};

use crate::core::{
    context::{ContextTracker, TrackingIndex},
    dirty::DirtyReceive,
    pipes::{self, Receiver, Sender},
    relation::RelationInner,
    Op_, Relation,
};

pub struct Save<C: Op_> {
    inner: Saved<C>,
    receiver: Receiver<Rc<Vec<C::T>>>,
}

struct SaveInner<C: Op_> {
    inner: RelationInner<C>,
    senders: Vec<Sender<Rc<Vec<C::T>>>>,
    dirty: DirtyReceive,
}

pub struct Saved<C: Op_> {
    context_tracker: ContextTracker,
    inner: Rc<RefCell<SaveInner<C>>>,
    tracking_index: TrackingIndex,
}

impl<C: Op_> Clone for Saved<C> {
    fn clone(&self) -> Self {
        Self {
            context_tracker: self.context_tracker.clone(),
            inner: Rc::clone(&self.inner),
            tracking_index: self.tracking_index,
        }
    }
}

impl<C: Op_> Saved<C> {
    pub fn new(rel: Relation<C>) -> Self {
        Self {
            context_tracker: rel.context_tracker,
            inner: Rc::new(RefCell::new(SaveInner {
                inner: rel.inner,
                senders: Vec::new(),
                dirty: rel.dirty.into_receive(),
            })),
            tracking_index: rel.tracking_index,
        }
    }
    pub fn get_shown(&self) -> Relation<Save<C>>
    where
        C::T: Clone,
    {
        let (sender, receiver) = pipes::new();
        let dirty = {
            let mut inner = self.inner.borrow_mut();
            inner.senders.push(sender);
            inner.dirty.add_target()
        };
        self.context_tracker.clone().add_relation(
            dirty,
            Save {
                inner: self.clone(),
                receiver,
            },
            vec![self.tracking_index],
        )
    }
    pub(super) fn borrow(&self) -> Ref<RelationInner<C>> {
        Ref::map(self.inner.borrow(), |x| &x.inner)
    }
    pub(super) fn propagate(&mut self) {
        if self.inner.borrow_mut().dirty.take_status() {
            let data = Rc::new(self.inner.borrow_mut().inner.get_vec());
            for sender in self.inner.borrow_mut().senders.iter_mut() {
                sender.send(Rc::clone(&data))
            }
        }
    }
}

impl<C: Op_> Op_ for Save<C>
where
    C::T: Clone,
{
    type T = C::T;

    fn foreach<'a>(&'a mut self, mut continuation: impl FnMut(Self::T) + 'a) {
        self.inner.propagate();
        for data in self.receiver.receive() {
            for x in data.iter() {
                continuation(x.clone())
            }
        }
    }

    fn get_type_name() -> &'static str {
        "save"
    }
}

impl<C: Op_> Relation<C> {
    pub fn save(self) -> Saved<C>
    where
        C::T: Clone,
    {
        Saved::new(self)
    }
}

impl<C: Op_> Clone for Relation<Save<C>>
where
    C::T: Clone,
{
    fn clone(&self) -> Self {
        self.inner.inner.inner.get()
    }
}