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
use std::borrow::Borrow;
use std::cell::RefMut;
use std::fmt;
use std::ops::Deref;
use std::ops::DerefMut;

/// Kind of like Cow, but might also be a RefMut (from a RefCell)
/// Also, does not use ToOwned.
/// This is because the main point of an XRefMut is not to Copy On Write,
/// but to either hold an owned value or a reference to a value
pub enum XRefMut<'a, T> {
    Borrowed(&'a mut T),
    Ref(RefMut<'a, T>),
    Owned(T),
}

impl<'a, T> XRefMut<'a, T> {
    pub fn to_ref(&self) -> &T {
        match self {
            Self::Borrowed(r) => r,
            Self::Ref(r) => r,
            Self::Owned(t) => t,
        }
    }
    pub fn to_mut(&mut self) -> &mut T {
        match self {
            Self::Borrowed(r) => r,
            Self::Ref(r) => r,
            Self::Owned(ref mut t) => t,
        }
    }
}

impl<'a, T: Clone> XRefMut<'a, T> {
    /// Extracts the owned data.
    /// Clones the data if it is not already owned.
    pub fn into_owned(self) -> T {
        match self {
            Self::Borrowed(r) => r.clone(),
            Self::Ref(r) => r.clone(),
            Self::Owned(t) => t,
        }
    }
}

impl<'a, T> AsRef<T> for XRefMut<'a, T> {
    fn as_ref(&self) -> &T {
        match self {
            Self::Borrowed(r) => r,
            Self::Ref(r) => r,
            Self::Owned(t) => t.borrow(),
        }
    }
}

impl<'a, T> Borrow<T> for XRefMut<'a, T> {
    fn borrow(&self) -> &T {
        &**self
    }
}

impl<'a, T: fmt::Debug> fmt::Debug for XRefMut<'a, T> {
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        match self {
            Self::Borrowed(r) => fmt::Debug::fmt(r, f),
            Self::Ref(r) => fmt::Debug::fmt(r, f),
            Self::Owned(t) => fmt::Debug::fmt(t, f),
        }
    }
}

impl<'a, T: fmt::Display> fmt::Display for XRefMut<'a, T> {
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        match self {
            Self::Borrowed(r) => fmt::Display::fmt(r, f),
            Self::Ref(r) => fmt::Display::fmt(r, f),
            Self::Owned(t) => fmt::Display::fmt(t, f),
        }
    }
}

impl<'a, T: Default> Default for XRefMut<'a, T> {
    fn default() -> Self {
        Self::Owned(Default::default())
    }
}

impl<'a, T> Deref for XRefMut<'a, T> {
    type Target = T;
    fn deref(&self) -> &T {
        self.to_ref()
    }
}

impl<'a, T> DerefMut for XRefMut<'a, T> {
    fn deref_mut(&mut self) -> &mut T {
        self.to_mut()
    }
}