fractal/session/model/room/
typing_list.rs

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
use gtk::{gio, glib, prelude::*, subclass::prelude::*};

use super::Member;

mod imp {
    use std::cell::{Cell, RefCell};

    use super::*;

    #[derive(Debug, Default, glib::Properties)]
    #[properties(wrapper_type = super::TypingList)]
    pub struct TypingList {
        /// The list of members currently typing.
        members: RefCell<Vec<Member>>,
        /// Whether this list is empty.
        #[property(get, explicit_notify)]
        is_empty: Cell<bool>,
    }

    #[glib::object_subclass]
    impl ObjectSubclass for TypingList {
        const NAME: &'static str = "TypingList";
        type Type = super::TypingList;
        type Interfaces = (gio::ListModel,);
    }

    #[glib::derived_properties]
    impl ObjectImpl for TypingList {}

    impl ListModelImpl for TypingList {
        fn item_type(&self) -> glib::Type {
            Member::static_type()
        }

        fn n_items(&self) -> u32 {
            self.members.borrow().len() as u32
        }

        fn item(&self, position: u32) -> Option<glib::Object> {
            self.members
                .borrow()
                .get(position as usize)
                .map(|member| member.clone().upcast())
        }
    }

    impl TypingList {
        /// Set whether the list is empty.
        fn set_is_empty(&self, is_empty: bool) {
            if self.is_empty.get() == is_empty {
                return;
            }

            self.is_empty.set(is_empty);
            self.obj().notify_is_empty();
        }

        /// Update this list with the given list of typing members.
        pub(super) fn update(&self, new_members: Vec<Member>) {
            if new_members.is_empty() {
                self.set_is_empty(true);

                return;
            }

            let (removed, added) = {
                let mut members = self.members.borrow_mut();
                let removed = members.len() as u32;
                let added = new_members.len() as u32;
                *members = new_members;
                (removed, added)
            };

            self.obj().items_changed(0, removed, added);
            self.set_is_empty(false);
        }
    }
}

glib::wrapper! {
    /// List of members that are currently typing.
    pub struct TypingList(ObjectSubclass<imp::TypingList>)
        @implements gio::ListModel;
}

impl TypingList {
    /// Construct a new empty `TypingList`.
    pub fn new() -> Self {
        glib::Object::new()
    }

    /// Update this list with the given list of typing members.
    pub(super) fn update(&self, new_members: Vec<Member>) {
        self.imp().update(new_members);
    }
}

impl Default for TypingList {
    fn default() -> Self {
        Self::new()
    }
}