matrix-sdk-ui 0.17.0

GUI-centric utilities on top of matrix-rust-sdk (experimental).
Documentation
// Copyright 2024 The Matrix.org Foundation C.I.C.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//     http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

use std::collections::HashSet;

use ruma::{
    OwnedEventId,
    events::{StateEventContentChange, room::pinned_events::RoomPinnedEventsEventContent},
};

#[derive(Clone, Debug)]
#[cfg_attr(feature = "uniffi", derive(uniffi::Enum))]
/// The type of change between the previous and current pinned events.
pub enum RoomPinnedEventsChange {
    /// Only new event ids were added.
    Added,
    /// Only event ids were removed.
    Removed,
    /// Some change other than only adding or only removing ids happened.
    Changed,
}

impl From<&StateEventContentChange<RoomPinnedEventsEventContent>> for RoomPinnedEventsChange {
    fn from(value: &StateEventContentChange<RoomPinnedEventsEventContent>) -> Self {
        match value {
            StateEventContentChange::Original { content, prev_content } => {
                if let Some(prev_content) = prev_content {
                    let mut new_pinned: HashSet<&OwnedEventId> =
                        HashSet::from_iter(&content.pinned);
                    if let Some(old_pinned) = &prev_content.pinned {
                        let mut still_pinned: HashSet<&OwnedEventId> =
                            HashSet::from_iter(old_pinned);

                        // Newly added elements will be kept in new_pinned, previous ones in
                        // still_pinned instead
                        still_pinned.retain(|item| new_pinned.remove(item));

                        let added = !new_pinned.is_empty();
                        let removed = still_pinned.len() < old_pinned.len();
                        if added && removed {
                            RoomPinnedEventsChange::Changed
                        } else if added {
                            RoomPinnedEventsChange::Added
                        } else if removed {
                            RoomPinnedEventsChange::Removed
                        } else {
                            // Any other case
                            RoomPinnedEventsChange::Changed
                        }
                    } else {
                        // We don't know the previous state, so let's assume a generic change
                        RoomPinnedEventsChange::Changed
                    }
                } else {
                    // If there is no previous content we can assume the first pinned event id was
                    // just added
                    RoomPinnedEventsChange::Added
                }
            }
            StateEventContentChange::Redacted(_) => RoomPinnedEventsChange::Changed,
        }
    }
}

#[cfg(test)]
mod tests {
    use assert_matches::assert_matches;
    use ruma::{
        events::{
            StateEventContentChange,
            room::pinned_events::{
                RedactedRoomPinnedEventsEventContent, RoomPinnedEventsEventContent,
            },
        },
        owned_event_id,
    };

    use crate::timeline::event_item::content::pinned_events::RoomPinnedEventsChange;

    #[test]
    fn redacted_pinned_events_content_has_generic_changes() {
        let content =
            StateEventContentChange::Redacted(RedactedRoomPinnedEventsEventContent::new());
        let ret: RoomPinnedEventsChange = (&content).into();
        assert_matches!(ret, RoomPinnedEventsChange::Changed);
    }

    #[test]
    fn pinned_events_content_with_no_prev_content_returns_added() {
        let content = StateEventContentChange::Original {
            content: RoomPinnedEventsEventContent::new(vec![owned_event_id!("$1")]),
            prev_content: None,
        };
        let ret: RoomPinnedEventsChange = (&content).into();
        assert_matches!(ret, RoomPinnedEventsChange::Added);
    }

    #[test]
    fn pinned_events_content_with_added_ids_returns_added() {
        let content = StateEventContentChange::Original {
            content: RoomPinnedEventsEventContent::new(vec![owned_event_id!("$1")]),
            prev_content: Some(RoomPinnedEventsEventContent::new(Vec::new()).into()),
        };
        let ret: RoomPinnedEventsChange = (&content).into();
        assert_matches!(ret, RoomPinnedEventsChange::Added);
    }

    #[test]
    fn pinned_events_content_with_removed_ids_returns_removed() {
        let content = StateEventContentChange::Original {
            content: RoomPinnedEventsEventContent::new(Vec::new()),
            prev_content: Some(
                RoomPinnedEventsEventContent::new(vec![owned_event_id!("$1")]).into(),
            ),
        };
        let ret: RoomPinnedEventsChange = (&content).into();
        assert_matches!(ret, RoomPinnedEventsChange::Removed);
    }

    #[test]
    fn pinned_events_content_with_added_and_removed_ids_returns_changed() {
        let content = StateEventContentChange::Original {
            content: RoomPinnedEventsEventContent::new(vec![owned_event_id!("$2")]),
            prev_content: Some(
                RoomPinnedEventsEventContent::new(vec![owned_event_id!("$1")]).into(),
            ),
        };
        let ret: RoomPinnedEventsChange = (&content).into();
        assert_matches!(ret, RoomPinnedEventsChange::Changed);
    }

    #[test]
    fn pinned_events_content_with_changed_order_returns_changed() {
        let content = StateEventContentChange::Original {
            content: RoomPinnedEventsEventContent::new(vec![
                owned_event_id!("$2"),
                owned_event_id!("$1"),
            ]),
            prev_content: Some(
                RoomPinnedEventsEventContent::new(vec![
                    owned_event_id!("$1"),
                    owned_event_id!("$2"),
                ])
                .into(),
            ),
        };
        let ret: RoomPinnedEventsChange = (&content).into();
        assert_matches!(ret, RoomPinnedEventsChange::Changed);
    }

    #[test]
    fn pinned_events_content_with_no_changes_returns_changed() {
        // Returning Changed is counter-intuitive, but it makes no sense to display in
        // the timeline 'UserFoo didn't change anything in the pinned events'

        let content = StateEventContentChange::Original {
            content: RoomPinnedEventsEventContent::new(vec![
                owned_event_id!("$1"),
                owned_event_id!("$2"),
            ]),
            prev_content: Some(
                RoomPinnedEventsEventContent::new(vec![
                    owned_event_id!("$1"),
                    owned_event_id!("$2"),
                ])
                .into(),
            ),
        };
        let ret: RoomPinnedEventsChange = (&content).into();
        assert_matches!(ret, RoomPinnedEventsChange::Changed);
    }
}