matrix_sdk_ui/timeline/event_item/content/
message.rs1use std::fmt;
18
19use ruma::{
20 events::{
21 poll::unstable_start::{
22 NewUnstablePollStartEventContentWithoutRelation, SyncUnstablePollStartEvent,
23 UnstablePollStartEventContent,
24 },
25 room::message::{
26 MessageType, Relation, RoomMessageEventContent, RoomMessageEventContentWithoutRelation,
27 SyncRoomMessageEvent,
28 },
29 AnySyncMessageLikeEvent, AnySyncTimelineEvent, BundledMessageLikeRelations, Mentions,
30 },
31 html::RemoveReplyFallback,
32 serde::Raw,
33};
34use tracing::{error, trace};
35
36use crate::DEFAULT_SANITIZER_MODE;
37
38#[derive(Clone)]
40pub struct Message {
41 pub(in crate::timeline) msgtype: MessageType,
42 pub(in crate::timeline) edited: bool,
43 pub(in crate::timeline) mentions: Option<Mentions>,
44}
45
46impl Message {
47 pub(in crate::timeline) fn from_event(
49 c: RoomMessageEventContent,
50 edit: Option<RoomMessageEventContentWithoutRelation>,
51 remove_reply_fallback: RemoveReplyFallback,
52 ) -> Self {
53 let mut msgtype = c.msgtype;
54 msgtype.sanitize(DEFAULT_SANITIZER_MODE, remove_reply_fallback);
55
56 let mut ret = Self { msgtype, edited: false, mentions: c.mentions };
57
58 if let Some(edit) = edit {
59 ret.apply_edit(edit);
60 }
61
62 ret
63 }
64
65 pub(crate) fn apply_edit(&mut self, mut new_content: RoomMessageEventContentWithoutRelation) {
67 trace!("applying edit to a Message");
68 new_content.msgtype.sanitize(DEFAULT_SANITIZER_MODE, RemoveReplyFallback::No);
70 self.msgtype = new_content.msgtype;
71 self.mentions = new_content.mentions;
72 self.edited = true;
73 }
74
75 pub fn msgtype(&self) -> &MessageType {
77 &self.msgtype
78 }
79
80 pub fn body(&self) -> &str {
84 self.msgtype.body()
85 }
86
87 pub fn is_edited(&self) -> bool {
90 self.edited
91 }
92
93 pub fn mentions(&self) -> Option<&Mentions> {
95 self.mentions.as_ref()
96 }
97}
98
99pub(crate) fn extract_bundled_edit_event_json(
105 raw: &Raw<AnySyncTimelineEvent>,
106) -> Option<Raw<AnySyncTimelineEvent>> {
107 let raw_unsigned: Raw<serde_json::Value> = raw.get_field("unsigned").ok()??;
109 let raw_relations: Raw<serde_json::Value> = raw_unsigned.get_field("m.relations").ok()??;
110 raw_relations.get_field::<Raw<AnySyncTimelineEvent>>("m.replace").ok()?
111}
112
113pub(crate) fn extract_room_msg_edit_content(
116 relations: BundledMessageLikeRelations<AnySyncMessageLikeEvent>,
117) -> Option<RoomMessageEventContentWithoutRelation> {
118 match *relations.replace? {
119 AnySyncMessageLikeEvent::RoomMessage(SyncRoomMessageEvent::Original(ev)) => match ev
120 .content
121 .relates_to
122 {
123 Some(Relation::Replacement(re)) => {
124 trace!("found a bundled edit event in a room message");
125 Some(re.new_content)
126 }
127 _ => {
128 error!("got m.room.message event with an edit without a valid m.replace relation");
129 None
130 }
131 },
132
133 AnySyncMessageLikeEvent::RoomMessage(SyncRoomMessageEvent::Redacted(_)) => None,
134
135 _ => {
136 error!("got m.room.message event with an edit of a different event type");
137 None
138 }
139 }
140}
141
142pub(crate) fn extract_poll_edit_content(
145 relations: BundledMessageLikeRelations<AnySyncMessageLikeEvent>,
146) -> Option<NewUnstablePollStartEventContentWithoutRelation> {
147 match *relations.replace? {
148 AnySyncMessageLikeEvent::UnstablePollStart(SyncUnstablePollStartEvent::Original(ev)) => {
149 match ev.content {
150 UnstablePollStartEventContent::Replacement(re) => {
151 trace!("found a bundled edit event in a poll");
152 Some(re.relates_to.new_content)
153 }
154 _ => {
155 error!("got new poll start event in a bundled edit");
156 None
157 }
158 }
159 }
160
161 AnySyncMessageLikeEvent::UnstablePollStart(SyncUnstablePollStartEvent::Redacted(_)) => None,
162
163 _ => {
164 error!("got poll edit event with an edit of a different event type");
165 None
166 }
167 }
168}
169
170#[cfg(not(tarpaulin_include))]
171impl fmt::Debug for Message {
172 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
173 let Self { msgtype: _, edited, mentions: _ } = self;
174 f.debug_struct("Message").field("edited", edited).finish_non_exhaustive()
177 }
178}