1use serde::Serialize;
23use super::{
4 AddMentions, ForwardThread, MessageType, Relation, ReplacementMetadata, ReplyMetadata,
5 ReplyWithinThread, RoomMessageEventContent,
6};
7use crate::{
8 relation::{InReplyTo, Replacement, Thread},
9 Mentions,
10};
1112/// Form of [`RoomMessageEventContent`] without relation.
13#[derive(Clone, Debug, Serialize)]
14#[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
15pub struct RoomMessageEventContentWithoutRelation {
16/// A key which identifies the type of message being sent.
17 ///
18 /// This also holds the specific content of each message.
19#[serde(flatten)]
20pub msgtype: MessageType,
2122/// The [mentions] of this event.
23 ///
24 /// [mentions]: https://spec.matrix.org/latest/client-server-api/#user-and-room-mentions
25#[serde(rename = "m.mentions", skip_serializing_if = "Option::is_none")]
26pub mentions: Option<Mentions>,
27}
2829impl RoomMessageEventContentWithoutRelation {
30/// Creates a new `RoomMessageEventContentWithoutRelation` with the given `MessageType`.
31pub fn new(msgtype: MessageType) -> Self {
32Self { msgtype, mentions: None }
33 }
3435/// A constructor to create a plain text message.
36pub fn text_plain(body: impl Into<String>) -> Self {
37Self::new(MessageType::text_plain(body))
38 }
3940/// A constructor to create an html message.
41pub fn text_html(body: impl Into<String>, html_body: impl Into<String>) -> Self {
42Self::new(MessageType::text_html(body, html_body))
43 }
4445/// A constructor to create a markdown message.
46#[cfg(feature = "markdown")]
47pub fn text_markdown(body: impl AsRef<str> + Into<String>) -> Self {
48Self::new(MessageType::text_markdown(body))
49 }
5051/// A constructor to create a plain text notice.
52pub fn notice_plain(body: impl Into<String>) -> Self {
53Self::new(MessageType::notice_plain(body))
54 }
5556/// A constructor to create an html notice.
57pub fn notice_html(body: impl Into<String>, html_body: impl Into<String>) -> Self {
58Self::new(MessageType::notice_html(body, html_body))
59 }
6061/// A constructor to create a markdown notice.
62#[cfg(feature = "markdown")]
63pub fn notice_markdown(body: impl AsRef<str> + Into<String>) -> Self {
64Self::new(MessageType::notice_markdown(body))
65 }
6667/// A constructor to create a plain text emote.
68pub fn emote_plain(body: impl Into<String>) -> Self {
69Self::new(MessageType::emote_plain(body))
70 }
7172/// A constructor to create an html emote.
73pub fn emote_html(body: impl Into<String>, html_body: impl Into<String>) -> Self {
74Self::new(MessageType::emote_html(body, html_body))
75 }
7677/// A constructor to create a markdown emote.
78#[cfg(feature = "markdown")]
79pub fn emote_markdown(body: impl AsRef<str> + Into<String>) -> Self {
80Self::new(MessageType::emote_markdown(body))
81 }
8283/// Transform `self` into a `RoomMessageEventContent` with the given relation.
84pub fn with_relation(
85self,
86 relates_to: Option<Relation<RoomMessageEventContentWithoutRelation>>,
87 ) -> RoomMessageEventContent {
88let Self { msgtype, mentions } = self;
89 RoomMessageEventContent { msgtype, relates_to, mentions }
90 }
9192/// Turns `self` into a [rich reply] to the message using the given metadata.
93 ///
94 /// Sets the `in_reply_to` field inside `relates_to`, and optionally the `rel_type` to
95 /// `m.thread` if the metadata has a `thread` and `ForwardThread::Yes` is used.
96 ///
97 /// If `AddMentions::Yes` is used, the `sender` in the metadata is added as a user mention.
98 ///
99 /// [rich reply]: https://spec.matrix.org/latest/client-server-api/#rich-replies
100#[track_caller]
101pub fn make_reply_to<'a>(
102mut self,
103 metadata: impl Into<ReplyMetadata<'a>>,
104 forward_thread: ForwardThread,
105 add_mentions: AddMentions,
106 ) -> RoomMessageEventContent {
107let metadata = metadata.into();
108let original_event_id = metadata.event_id.to_owned();
109110let original_thread_id = metadata
111 .thread
112 .filter(|_| forward_thread == ForwardThread::Yes)
113 .map(|thread| thread.event_id.clone());
114let relates_to = if let Some(event_id) = original_thread_id {
115 Relation::Thread(Thread::plain(event_id.to_owned(), original_event_id.to_owned()))
116 } else {
117 Relation::Reply { in_reply_to: InReplyTo { event_id: original_event_id.to_owned() } }
118 };
119120if add_mentions == AddMentions::Yes {
121self.mentions
122 .get_or_insert_with(Mentions::new)
123 .user_ids
124 .insert(metadata.sender.to_owned());
125 }
126127self.with_relation(Some(relates_to))
128 }
129130/// Turns `self` into a new message for a [thread], that is optionally a reply.
131 ///
132 /// Looks for the `thread` in the given metadata. If it exists, this message will be in the same
133 /// thread. If it doesn't, a new thread is created with the `event_id` in the metadata as the
134 /// root.
135 ///
136 /// It also sets the `in_reply_to` field inside `relates_to` to point the `event_id`
137 /// in the metadata. If `ReplyWithinThread::Yes` is used, the metadata should be constructed
138 /// from the event to make a reply to, otherwise it should be constructed from the latest
139 /// event in the thread.
140 ///
141 /// If `AddMentions::Yes` is used, the `sender` in the metadata is added as a user mention.
142 ///
143 /// [thread]: https://spec.matrix.org/latest/client-server-api/#threading
144pub fn make_for_thread<'a>(
145self,
146 metadata: impl Into<ReplyMetadata<'a>>,
147 is_reply: ReplyWithinThread,
148 add_mentions: AddMentions,
149 ) -> RoomMessageEventContent {
150let metadata = metadata.into();
151152let mut content = if is_reply == ReplyWithinThread::Yes {
153self.make_reply_to(metadata, ForwardThread::No, add_mentions)
154 } else {
155self.into()
156 };
157158let thread_root = if let Some(Thread { event_id, .. }) = &metadata.thread {
159 event_id.to_owned()
160 } else {
161 metadata.event_id.to_owned()
162 };
163164 content.relates_to = Some(Relation::Thread(Thread {
165 event_id: thread_root,
166 in_reply_to: Some(InReplyTo { event_id: metadata.event_id.to_owned() }),
167 is_falling_back: is_reply == ReplyWithinThread::No,
168 }));
169170 content
171 }
172173/// Turns `self` into a [replacement] (or edit) for a given message.
174 ///
175 /// The first argument after `self` can be `&OriginalRoomMessageEvent` or
176 /// `&OriginalSyncRoomMessageEvent` if you don't want to create `ReplacementMetadata` separately
177 /// before calling this function.
178 ///
179 /// This takes the content and sets it in `m.new_content`, and modifies the `content` to include
180 /// a fallback.
181 ///
182 /// If this message contains [`Mentions`], they are copied into `m.new_content` to keep the same
183 /// mentions, but the ones in `content` are filtered with the ones in the
184 /// [`ReplacementMetadata`] so only new mentions will trigger a notification.
185 ///
186 /// # Panics
187 ///
188 /// Panics if `self` has a `formatted_body` with a format other than HTML.
189 ///
190 /// [replacement]: https://spec.matrix.org/latest/client-server-api/#event-replacements
191#[track_caller]
192pub fn make_replacement(
193mut self,
194 metadata: impl Into<ReplacementMetadata>,
195 ) -> RoomMessageEventContent {
196let metadata = metadata.into();
197198let mentions = self.mentions.take();
199200// Only set mentions that were not there before.
201if let Some(mentions) = &mentions {
202let new_mentions = metadata
203 .mentions
204 .map(|old_mentions| {
205let mut new_mentions = Mentions::new();
206207 new_mentions.user_ids = mentions
208 .user_ids
209 .iter()
210 .filter(|u| !old_mentions.user_ids.contains(*u))
211 .cloned()
212 .collect();
213214 new_mentions.room = mentions.room && !old_mentions.room;
215216 new_mentions
217 })
218 .unwrap_or_else(|| mentions.clone());
219220self.mentions = Some(new_mentions);
221 };
222223// Prepare relates_to with the untouched msgtype.
224let relates_to = Relation::Replacement(Replacement {
225 event_id: metadata.event_id,
226 new_content: RoomMessageEventContentWithoutRelation {
227 msgtype: self.msgtype.clone(),
228 mentions,
229 },
230 });
231232self.msgtype.make_replacement_body();
233234let mut content = RoomMessageEventContent::from(self);
235 content.relates_to = Some(relates_to);
236237 content
238 }
239240/// Add the given [mentions] to this event.
241 ///
242 /// If no [`Mentions`] was set on this events, this sets it. Otherwise, this updates the current
243 /// mentions by extending the previous `user_ids` with the new ones, and applies a logical OR to
244 /// the values of `room`.
245 ///
246 /// [mentions]: https://spec.matrix.org/latest/client-server-api/#user-and-room-mentions
247pub fn add_mentions(mut self, mentions: Mentions) -> Self {
248self.mentions.get_or_insert_with(Mentions::new).add(mentions);
249self
250}
251}
252253impl From<MessageType> for RoomMessageEventContentWithoutRelation {
254fn from(msgtype: MessageType) -> Self {
255Self::new(msgtype)
256 }
257}
258259impl From<RoomMessageEventContent> for RoomMessageEventContentWithoutRelation {
260fn from(value: RoomMessageEventContent) -> Self {
261let RoomMessageEventContent { msgtype, mentions, .. } = value;
262Self { msgtype, mentions }
263 }
264}
265266impl From<RoomMessageEventContentWithoutRelation> for RoomMessageEventContent {
267fn from(value: RoomMessageEventContentWithoutRelation) -> Self {
268let RoomMessageEventContentWithoutRelation { msgtype, mentions } = value;
269Self { msgtype, relates_to: None, mentions }
270 }
271}