matrix_sdk_ui/timeline/mod.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 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937
// Copyright 2022 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.
//! A high-level view into a room's contents.
//!
//! See [`Timeline`] for details.
use std::{fs, path::PathBuf, pin::Pin, sync::Arc, task::Poll};
use algorithms::rfind_event_by_item_id;
use event_item::{extract_room_msg_edit_content, TimelineItemHandle};
use eyeball_im::VectorDiff;
use futures_core::Stream;
use imbl::Vector;
use matrix_sdk::{
attachment::AttachmentConfig,
event_cache::{EventCacheDropHandles, RoomEventCache},
event_handler::EventHandlerHandle,
executor::JoinHandle,
room::{edit::EditedContent, Receipts, Room},
send_queue::{RoomSendQueueError, SendHandle},
Client, Result,
};
use mime::Mime;
use pin_project_lite::pin_project;
use ruma::{
api::client::receipt::create_receipt::v3::ReceiptType,
events::{
poll::unstable_start::{NewUnstablePollStartEventContent, UnstablePollStartEventContent},
receipt::{Receipt, ReceiptThread},
room::{
message::{
AddMentions, ForwardThread, OriginalRoomMessageEvent,
RoomMessageEventContentWithoutRelation,
},
pinned_events::RoomPinnedEventsEventContent,
},
AnyMessageLikeEventContent, AnySyncMessageLikeEvent, AnySyncTimelineEvent,
SyncMessageLikeEvent,
},
serde::Raw,
EventId, MilliSecondsSinceUnixEpoch, OwnedEventId, OwnedUserId, RoomVersionId, UserId,
};
use thiserror::Error;
use tracing::{error, instrument, trace, warn};
use self::{
algorithms::rfind_event_by_id, controller::TimelineController, futures::SendAttachment,
};
use crate::timeline::pinned_events_loader::PinnedEventsRoom;
mod algorithms;
mod builder;
mod controller;
mod date_dividers;
mod error;
mod event_handler;
mod event_item;
pub mod event_type_filter;
pub mod futures;
mod item;
mod pagination;
mod pinned_events_loader;
mod reactions;
#[cfg(test)]
mod tests;
mod to_device;
mod traits;
mod virtual_item;
pub use self::{
builder::TimelineBuilder,
controller::default_event_filter,
error::*,
event_item::{
AnyOtherFullStateEventContent, EncryptedMessage, EventItemOrigin, EventSendState,
EventTimelineItem, InReplyToDetails, MemberProfileChange, MembershipChange, Message,
OtherState, PollResult, PollState, Profile, ReactionInfo, ReactionStatus,
ReactionsByKeyBySender, RepliedToEvent, RoomMembershipChange, RoomPinnedEventsChange,
Sticker, TimelineDetails, TimelineEventItemId, TimelineItemContent,
},
event_type_filter::TimelineEventTypeFilter,
item::{TimelineItem, TimelineItemKind, TimelineUniqueId},
pagination::LiveBackPaginationStatus,
traits::RoomExt,
virtual_item::VirtualTimelineItem,
};
/// Information needed to reply to an event.
#[derive(Debug, Clone)]
pub struct RepliedToInfo {
/// The event ID of the event to reply to.
event_id: OwnedEventId,
/// The sender of the event to reply to.
sender: OwnedUserId,
/// The timestamp of the event to reply to.
timestamp: MilliSecondsSinceUnixEpoch,
/// The content of the event to reply to.
content: ReplyContent,
}
impl RepliedToInfo {
/// The event ID of the event to reply to.
pub fn event_id(&self) -> &EventId {
&self.event_id
}
/// The sender of the event to reply to.
pub fn sender(&self) -> &UserId {
&self.sender
}
/// The content of the event to reply to.
pub fn content(&self) -> &ReplyContent {
&self.content
}
}
/// The content of a reply.
#[derive(Debug, Clone)]
pub enum ReplyContent {
/// Content of a message event.
Message(Message),
/// Content of any other kind of event stored as raw JSON.
Raw(Raw<AnySyncTimelineEvent>),
}
/// A high-level view into a regular¹ room's contents.
///
/// ¹ This type is meant to be used in the context of rooms without a
/// `room_type`, that is rooms that are primarily used to exchange text
/// messages.
#[derive(Debug)]
pub struct Timeline {
/// Cloneable, inner fields of the `Timeline`, shared with some background
/// tasks.
controller: TimelineController,
/// The event cache specialized for this room's view.
event_cache: RoomEventCache,
/// References to long-running tasks held by the timeline.
drop_handle: Arc<TimelineDropHandle>,
}
/// What should the timeline focus on?
#[derive(Clone, Debug, PartialEq)]
pub enum TimelineFocus {
/// Focus on live events, i.e. receive events from sync and append them in
/// real-time.
Live,
/// Focus on a specific event, e.g. after clicking a permalink.
Event { target: OwnedEventId, num_context_events: u16 },
/// Only show pinned events.
PinnedEvents { max_events_to_load: u16, max_concurrent_requests: u16 },
}
impl TimelineFocus {
pub(super) fn debug_string(&self) -> String {
match self {
TimelineFocus::Live => "live".to_owned(),
TimelineFocus::Event { target, .. } => format!("permalink:{target}"),
TimelineFocus::PinnedEvents { .. } => "pinned-events".to_owned(),
}
}
}
/// Changes how dividers get inserted, either in between each day or in between
/// each month
#[derive(Debug, Clone)]
pub enum DateDividerMode {
Daily,
Monthly,
}
impl Timeline {
/// Create a new [`TimelineBuilder`] for the given room.
pub fn builder(room: &Room) -> TimelineBuilder {
TimelineBuilder::new(room)
}
/// Returns the room for this timeline.
pub fn room(&self) -> &Room {
self.controller.room()
}
/// Clear all timeline items.
pub async fn clear(&self) {
self.controller.clear().await;
}
/// Retry decryption of previously un-decryptable events given a list of
/// session IDs whose keys have been imported.
///
/// # Examples
///
/// ```no_run
/// # use std::{path::PathBuf, time::Duration};
/// # use matrix_sdk::{Client, config::SyncSettings, ruma::room_id};
/// # use matrix_sdk_ui::Timeline;
/// # async {
/// # let mut client: Client = todo!();
/// # let room_id = ruma::room_id!("!example:example.org");
/// # let timeline: Timeline = todo!();
/// let path = PathBuf::from("/home/example/e2e-keys.txt");
/// let result =
/// client.encryption().import_room_keys(path, "secret-passphrase").await?;
///
/// // Given a timeline for a specific room_id
/// if let Some(keys_for_users) = result.keys.get(room_id) {
/// let session_ids = keys_for_users.values().flatten();
/// timeline.retry_decryption(session_ids).await;
/// }
/// # anyhow::Ok(()) };
/// ```
pub async fn retry_decryption<S: Into<String>>(
&self,
session_ids: impl IntoIterator<Item = S>,
) {
self.controller
.retry_event_decryption(
self.room(),
Some(session_ids.into_iter().map(Into::into).collect()),
)
.await;
}
#[tracing::instrument(skip(self))]
async fn retry_decryption_for_all_events(&self) {
self.controller.retry_event_decryption(self.room(), None).await;
}
/// Get the current timeline item for the given event ID, if any.
///
/// Will return a remote event, *or* a local echo that has been sent but not
/// yet replaced by a remote echo.
///
/// It's preferable to store the timeline items in the model for your UI, if
/// possible, instead of just storing IDs and coming back to the timeline
/// object to look up items.
pub async fn item_by_event_id(&self, event_id: &EventId) -> Option<EventTimelineItem> {
let items = self.controller.items().await;
let (_, item) = rfind_event_by_id(&items, event_id)?;
Some(item.to_owned())
}
/// Get the latest of the timeline's event items.
pub async fn latest_event(&self) -> Option<EventTimelineItem> {
if self.controller.is_live().await {
self.controller.items().await.last()?.as_event().cloned()
} else {
None
}
}
/// Get the current timeline items, and a stream of changes.
///
/// You can poll this stream to receive updates. See
/// [`futures_util::StreamExt`] for a high-level API on top of [`Stream`].
pub async fn subscribe(
&self,
) -> (Vector<Arc<TimelineItem>>, impl Stream<Item = VectorDiff<Arc<TimelineItem>>>) {
let (items, stream) = self.controller.subscribe().await;
let stream = TimelineStream::new(stream, self.drop_handle.clone());
(items, stream)
}
/// Get the current timeline items, and a batched stream of changes.
///
/// In contrast to [`subscribe`](Self::subscribe), this stream can yield
/// multiple diffs at once. The batching is done such that no arbitrary
/// delays are added.
pub async fn subscribe_batched(
&self,
) -> (Vector<Arc<TimelineItem>>, impl Stream<Item = Vec<VectorDiff<Arc<TimelineItem>>>>) {
let (items, stream) = self.controller.subscribe_batched().await;
let stream = TimelineStream::new(stream, self.drop_handle.clone());
(items, stream)
}
/// Send a message to the room, and add it to the timeline as a local echo.
///
/// For simplicity, this method doesn't currently allow custom message
/// types.
///
/// If the encryption feature is enabled, this method will transparently
/// encrypt the room message if the room is encrypted.
///
/// If sending the message fails, the local echo item will change its
/// `send_state` to [`EventSendState::SendingFailed`].
///
/// # Arguments
///
/// * `content` - The content of the message event.
///
/// [`MessageLikeUnsigned`]: ruma::events::MessageLikeUnsigned
/// [`SyncMessageLikeEvent`]: ruma::events::SyncMessageLikeEvent
#[instrument(skip(self, content), fields(room_id = ?self.room().room_id()))]
pub async fn send(
&self,
content: AnyMessageLikeEventContent,
) -> Result<SendHandle, RoomSendQueueError> {
self.room().send_queue().send(content).await
}
/// Send a reply to the given event.
///
/// Currently it only supports events with an event ID and JSON being
/// available (which can be removed by local redactions). This is subject to
/// change. Please check [`EventTimelineItem::can_be_replied_to`] to decide
/// whether to render a reply button.
///
/// The sender will be added to the mentions of the reply if
/// and only if the event has not been written by the sender.
///
/// # Arguments
///
/// * `content` - The content of the reply
///
/// * `replied_to_info` - A wrapper that contains the event ID, sender,
/// content and timestamp of the event to reply to
///
/// * `forward_thread` - Usually `Yes`, unless you explicitly want to the
/// reply to show up in the main timeline even though the `reply_item` is
/// part of a thread
#[instrument(skip(self, content, replied_to_info))]
pub async fn send_reply(
&self,
content: RoomMessageEventContentWithoutRelation,
replied_to_info: RepliedToInfo,
forward_thread: ForwardThread,
) -> Result<(), RoomSendQueueError> {
let event_id = replied_to_info.event_id;
// [The specification](https://spec.matrix.org/v1.10/client-server-api/#user-and-room-mentions) says:
//
// > Users should not add their own Matrix ID to the `m.mentions` property as
// > outgoing messages cannot self-notify.
//
// If the replied to event has been written by the current user, let's toggle to
// `AddMentions::No`.
let mention_the_sender = if self.room().own_user_id() == replied_to_info.sender {
AddMentions::No
} else {
AddMentions::Yes
};
let content = match replied_to_info.content {
ReplyContent::Message(msg) => {
let event = OriginalRoomMessageEvent {
event_id: event_id.to_owned(),
sender: replied_to_info.sender,
origin_server_ts: replied_to_info.timestamp,
room_id: self.room().room_id().to_owned(),
content: msg.to_content(),
unsigned: Default::default(),
};
content.make_reply_to(&event, forward_thread, mention_the_sender)
}
ReplyContent::Raw(raw_event) => content.make_reply_to_raw(
&raw_event,
event_id.to_owned(),
self.room().room_id(),
forward_thread,
mention_the_sender,
),
};
self.send(content.into()).await?;
Ok(())
}
/// Get the information needed to reply to the event with the given ID.
pub async fn replied_to_info_from_event_id(
&self,
event_id: &EventId,
) -> Result<RepliedToInfo, UnsupportedReplyItem> {
if let Some(timeline_item) = self.item_by_event_id(event_id).await {
return timeline_item.replied_to_info();
}
let event = self.room().event(event_id, None).await.map_err(|error| {
error!("Failed to fetch event with ID {event_id} with error: {error}");
UnsupportedReplyItem::MissingEvent
})?;
let raw_sync_event = event.into_raw();
let sync_event = raw_sync_event.deserialize().map_err(|error| {
error!("Failed to deserialize event with ID {event_id} with error: {error}");
UnsupportedReplyItem::FailedToDeserializeEvent
})?;
let reply_content = match &sync_event {
AnySyncTimelineEvent::MessageLike(message_like_event) => {
if let AnySyncMessageLikeEvent::RoomMessage(SyncMessageLikeEvent::Original(
original_message,
)) = message_like_event
{
ReplyContent::Message(Message::from_event(
original_message.content.clone(),
extract_room_msg_edit_content(message_like_event.relations()),
&self.items().await,
))
} else {
ReplyContent::Raw(raw_sync_event)
}
}
AnySyncTimelineEvent::State(_) => return Err(UnsupportedReplyItem::StateEvent),
};
Ok(RepliedToInfo {
event_id: event_id.to_owned(),
sender: sync_event.sender().to_owned(),
timestamp: sync_event.origin_server_ts(),
content: reply_content,
})
}
/// Edit an event given its [`TimelineEventItemId`] and some new content.
///
/// Only supports events for which [`EventTimelineItem::is_editable()`]
/// returns `true`.
#[instrument(skip(self, new_content))]
pub async fn edit(
&self,
item_id: &TimelineEventItemId,
new_content: EditedContent,
) -> Result<(), Error> {
let items = self.items().await;
let Some((_pos, item)) = rfind_event_by_item_id(&items, item_id) else {
return Err(Error::EventNotInTimeline(item_id.clone()));
};
match item.handle() {
TimelineItemHandle::Remote(event_id) => {
let content = self
.room()
.make_edit_event(event_id, new_content)
.await
.map_err(EditError::RoomError)?;
self.send(content).await?;
Ok(())
}
TimelineItemHandle::Local(handle) => {
// Relations are filled by the editing code itself.
let new_content: AnyMessageLikeEventContent = match new_content {
EditedContent::RoomMessage(message) => {
if matches!(item.content, TimelineItemContent::Message(_)) {
AnyMessageLikeEventContent::RoomMessage(message.into())
} else {
return Err(EditError::ContentMismatch {
original: item.content.debug_string().to_owned(),
new: "a message".to_owned(),
}
.into());
}
}
EditedContent::PollStart { new_content, .. } => {
if matches!(item.content, TimelineItemContent::Poll(_)) {
AnyMessageLikeEventContent::UnstablePollStart(
UnstablePollStartEventContent::New(
NewUnstablePollStartEventContent::new(new_content),
),
)
} else {
return Err(EditError::ContentMismatch {
original: item.content.debug_string().to_owned(),
new: "a poll".to_owned(),
}
.into());
}
}
EditedContent::MediaCaption { caption, formatted_caption, mentions } => {
if handle
.edit_media_caption(caption, formatted_caption, mentions)
.await
.map_err(RoomSendQueueError::StorageError)?
{
return Ok(());
}
return Err(EditError::InvalidLocalEchoState.into());
}
};
if !handle.edit(new_content).await.map_err(RoomSendQueueError::StorageError)? {
return Err(EditError::InvalidLocalEchoState.into());
}
Ok(())
}
}
}
/// Toggle a reaction on an event.
///
/// Adds or redacts a reaction based on the state of the reaction at the
/// time it is called.
///
/// When redacting a previous reaction, the redaction reason is not set.
///
/// Ensures that only one reaction is sent at a time to avoid race
/// conditions and spamming the homeserver with requests.
pub async fn toggle_reaction(
&self,
item_id: &TimelineEventItemId,
reaction_key: &str,
) -> Result<(), Error> {
self.controller.toggle_reaction_local(item_id, reaction_key).await?;
Ok(())
}
/// Sends an attachment to the room.
///
/// It does not currently support local echoes.
///
/// If the encryption feature is enabled, this method will transparently
/// encrypt the room message if the room is encrypted.
///
/// The attachment and its optional thumbnail are stored in the media cache
/// and can be retrieved at any time, by calling
/// [`Media::get_media_content()`] with the `MediaSource` that can be found
/// in the corresponding `TimelineEventItem`, and using a
/// `MediaFormat::File`.
///
/// # Arguments
///
/// * `source` - The source of the attachment to send.
///
/// * `mime_type` - The attachment's mime type.
///
/// * `config` - An attachment configuration object containing details about
/// the attachment like a thumbnail, its size, duration etc.
///
/// [`Media::get_media_content()`]: matrix_sdk::Media::get_media_content
#[instrument(skip_all)]
pub fn send_attachment(
&self,
source: impl Into<AttachmentSource>,
mime_type: Mime,
config: AttachmentConfig,
) -> SendAttachment<'_> {
SendAttachment::new(self, source.into(), mime_type, config)
}
/// Redact an event given its [`TimelineEventItemId`] and an optional
/// reason.
pub async fn redact(
&self,
item_id: &TimelineEventItemId,
reason: Option<&str>,
) -> Result<(), Error> {
let items = self.items().await;
let Some((_pos, event)) = rfind_event_by_item_id(&items, item_id) else {
return Err(RedactError::ItemNotFound(item_id.clone()).into());
};
match event.handle() {
TimelineItemHandle::Remote(event_id) => {
self.room().redact(event_id, reason, None).await.map_err(RedactError::HttpError)?;
}
TimelineItemHandle::Local(handle) => {
if !handle.abort().await.map_err(RoomSendQueueError::StorageError)? {
return Err(RedactError::InvalidLocalEchoState.into());
}
}
}
Ok(())
}
/// Fetch unavailable details about the event with the given ID.
///
/// This method only works for IDs of remote [`EventTimelineItem`]s,
/// to prevent losing details when a local echo is replaced by its
/// remote echo.
///
/// This method tries to make all the requests it can. If an error is
/// encountered for a given request, it is forwarded with the
/// [`TimelineDetails::Error`] variant.
///
/// # Arguments
///
/// * `event_id` - The event ID of the event to fetch details for.
///
/// # Errors
///
/// Returns an error if the identifier doesn't match any event with a remote
/// echo in the timeline, or if the event is removed from the timeline
/// before all requests are handled.
#[instrument(skip(self), fields(room_id = ?self.room().room_id()))]
pub async fn fetch_details_for_event(&self, event_id: &EventId) -> Result<(), Error> {
self.controller.fetch_in_reply_to_details(event_id).await
}
/// Fetch all member events for the room this timeline is displaying.
///
/// If the full member list is not known, sender profiles are currently
/// likely not going to be available. This will be fixed in the future.
///
/// If fetching the members fails, any affected timeline items will have
/// the `sender_profile` set to [`TimelineDetails::Error`].
#[instrument(skip_all)]
pub async fn fetch_members(&self) {
self.controller.set_sender_profiles_pending().await;
match self.room().sync_members().await {
Ok(_) => {
self.controller.update_missing_sender_profiles().await;
}
Err(e) => {
self.controller.set_sender_profiles_error(Arc::new(e)).await;
}
}
}
/// Get the latest read receipt for the given user.
///
/// Contrary to [`Room::load_user_receipt()`] that only keeps track of read
/// receipts received from the homeserver, this keeps also track of implicit
/// read receipts in this timeline, i.e. when a room member sends an event.
#[instrument(skip(self))]
pub async fn latest_user_read_receipt(
&self,
user_id: &UserId,
) -> Option<(OwnedEventId, Receipt)> {
self.controller.latest_user_read_receipt(user_id).await
}
/// Get the ID of the timeline event with the latest read receipt for the
/// given user.
///
/// In contrary to [`Self::latest_user_read_receipt()`], this allows to know
/// the position of the read receipt in the timeline even if the event it
/// applies to is not visible in the timeline, unless the event is unknown
/// by this timeline.
#[instrument(skip(self))]
pub async fn latest_user_read_receipt_timeline_event_id(
&self,
user_id: &UserId,
) -> Option<OwnedEventId> {
self.controller.latest_user_read_receipt_timeline_event_id(user_id).await
}
/// Subscribe to changes in the read receipts of our own user.
pub async fn subscribe_own_user_read_receipts_changed(&self) -> impl Stream<Item = ()> {
self.controller.subscribe_own_user_read_receipts_changed().await
}
/// Send the given receipt.
///
/// This uses [`Room::send_single_receipt`] internally, but checks
/// first if the receipt points to an event in this timeline that is more
/// recent than the current ones, to avoid unnecessary requests.
///
/// Returns a boolean indicating if it sent the request or not.
#[instrument(skip(self), fields(room_id = ?self.room().room_id()))]
pub async fn send_single_receipt(
&self,
receipt_type: ReceiptType,
thread: ReceiptThread,
event_id: OwnedEventId,
) -> Result<bool> {
if !self.controller.should_send_receipt(&receipt_type, &thread, &event_id).await {
trace!(
"not sending receipt, because we already cover the event with a previous receipt"
);
return Ok(false);
}
trace!("sending receipt");
self.room().send_single_receipt(receipt_type, thread, event_id).await?;
Ok(true)
}
/// Send the given receipts.
///
/// This uses [`Room::send_multiple_receipts`] internally, but
/// checks first if the receipts point to events in this timeline that
/// are more recent than the current ones, to avoid unnecessary
/// requests.
#[instrument(skip(self))]
pub async fn send_multiple_receipts(&self, mut receipts: Receipts) -> Result<()> {
if let Some(fully_read) = &receipts.fully_read {
if !self
.controller
.should_send_receipt(
&ReceiptType::FullyRead,
&ReceiptThread::Unthreaded,
fully_read,
)
.await
{
receipts.fully_read = None;
}
}
if let Some(read_receipt) = &receipts.public_read_receipt {
if !self
.controller
.should_send_receipt(&ReceiptType::Read, &ReceiptThread::Unthreaded, read_receipt)
.await
{
receipts.public_read_receipt = None;
}
}
if let Some(private_read_receipt) = &receipts.private_read_receipt {
if !self
.controller
.should_send_receipt(
&ReceiptType::ReadPrivate,
&ReceiptThread::Unthreaded,
private_read_receipt,
)
.await
{
receipts.private_read_receipt = None;
}
}
self.room().send_multiple_receipts(receipts).await
}
/// Mark the room as read by sending an unthreaded read receipt on the
/// latest event, be it visible or not.
///
/// This works even if the latest event belongs to a thread, as a threaded
/// reply also belongs to the unthreaded timeline. No threaded receipt
/// will be sent here (see also #3123).
///
/// Returns a boolean indicating if we sent the request or not.
#[instrument(skip(self), fields(room_id = ?self.room().room_id()))]
pub async fn mark_as_read(&self, receipt_type: ReceiptType) -> Result<bool> {
if let Some(event_id) = self.controller.latest_event_id().await {
self.send_single_receipt(receipt_type, ReceiptThread::Unthreaded, event_id).await
} else {
trace!("can't mark room as read because there's no latest event id");
Ok(false)
}
}
/// Adds a new pinned event by sending an updated `m.room.pinned_events`
/// event containing the new event id.
///
/// This method will first try to get the pinned events from the current
/// room's state and if it fails to do so it'll try to load them from the
/// homeserver.
///
/// Returns `true` if we pinned the event, `false` if the event was already
/// pinned.
pub async fn pin_event(&self, event_id: &EventId) -> Result<bool> {
let mut pinned_event_ids = if let Some(event_ids) = self.room().pinned_event_ids() {
event_ids
} else {
self.room().load_pinned_events().await?.unwrap_or_default()
};
let event_id = event_id.to_owned();
if pinned_event_ids.contains(&event_id) {
Ok(false)
} else {
pinned_event_ids.push(event_id);
let content = RoomPinnedEventsEventContent::new(pinned_event_ids);
self.room().send_state_event(content).await?;
Ok(true)
}
}
/// Removes a pinned event by sending an updated `m.room.pinned_events`
/// event without the event id we want to remove.
///
/// This method will first try to get the pinned events from the current
/// room's state and if it fails to do so it'll try to load them from the
/// homeserver.
///
/// Returns `true` if we unpinned the event, `false` if the event wasn't
/// pinned before.
pub async fn unpin_event(&self, event_id: &EventId) -> Result<bool> {
let mut pinned_event_ids = if let Some(event_ids) = self.room().pinned_event_ids() {
event_ids
} else {
self.room().load_pinned_events().await?.unwrap_or_default()
};
let event_id = event_id.to_owned();
if let Some(idx) = pinned_event_ids.iter().position(|e| *e == *event_id) {
pinned_event_ids.remove(idx);
let content = RoomPinnedEventsEventContent::new(pinned_event_ids);
self.room().send_state_event(content).await?;
Ok(true)
} else {
Ok(false)
}
}
}
/// Test helpers, likely not very useful in production.
#[doc(hidden)]
impl Timeline {
/// Get the current list of timeline items.
pub async fn items(&self) -> Vector<Arc<TimelineItem>> {
self.controller.items().await
}
pub async fn subscribe_filter_map<U: Clone>(
&self,
f: impl Fn(Arc<TimelineItem>) -> Option<U>,
) -> (Vector<U>, impl Stream<Item = VectorDiff<U>>) {
let (items, stream) = self.controller.subscribe_filter_map(f).await;
let stream = TimelineStream::new(stream, self.drop_handle.clone());
(items, stream)
}
}
#[derive(Debug)]
struct TimelineDropHandle {
client: Client,
event_handler_handles: Vec<EventHandlerHandle>,
room_update_join_handle: JoinHandle<()>,
pinned_events_join_handle: Option<JoinHandle<()>>,
room_key_from_backups_join_handle: JoinHandle<()>,
room_keys_received_join_handle: JoinHandle<()>,
room_key_backup_enabled_join_handle: JoinHandle<()>,
local_echo_listener_handle: JoinHandle<()>,
_event_cache_drop_handle: Arc<EventCacheDropHandles>,
encryption_changes_handle: JoinHandle<()>,
}
impl Drop for TimelineDropHandle {
fn drop(&mut self) {
for handle in self.event_handler_handles.drain(..) {
self.client.remove_event_handler(handle);
}
if let Some(handle) = self.pinned_events_join_handle.take() {
handle.abort()
};
self.local_echo_listener_handle.abort();
self.room_update_join_handle.abort();
self.room_key_from_backups_join_handle.abort();
self.room_key_backup_enabled_join_handle.abort();
self.room_keys_received_join_handle.abort();
self.encryption_changes_handle.abort();
}
}
pin_project! {
struct TimelineStream<S> {
#[pin]
inner: S,
drop_handle: Arc<TimelineDropHandle>,
}
}
impl<S> TimelineStream<S> {
fn new(inner: S, drop_handle: Arc<TimelineDropHandle>) -> Self {
Self { inner, drop_handle }
}
}
impl<S: Stream> Stream for TimelineStream<S> {
type Item = S::Item;
fn poll_next(
self: Pin<&mut Self>,
cx: &mut std::task::Context<'_>,
) -> Poll<Option<Self::Item>> {
self.project().inner.poll_next(cx)
}
}
pub type TimelineEventFilterFn =
dyn Fn(&AnySyncTimelineEvent, &RoomVersionId) -> bool + Send + Sync;
/// A source for sending an attachment.
///
/// The [`AttachmentSource::File`] variant can be constructed from any type that
/// implements `Into<PathBuf>`.
#[derive(Debug, Clone)]
pub enum AttachmentSource {
/// The data of the attachment.
Data {
/// The bytes of the attachment.
bytes: Vec<u8>,
/// The filename of the attachment.
filename: String,
},
/// An attachment loaded from a file.
///
/// The bytes and the filename will be read from the file at the given path.
File(PathBuf),
}
impl AttachmentSource {
/// Try to convert this attachment source into a `(bytes, filename)` tuple.
pub(crate) fn try_into_bytes_and_filename(self) -> Result<(Vec<u8>, String), Error> {
match self {
Self::Data { bytes, filename } => Ok((bytes, filename)),
Self::File(path) => {
let filename = path
.file_name()
.ok_or(Error::InvalidAttachmentFileName)?
.to_str()
.ok_or(Error::InvalidAttachmentFileName)?
.to_owned();
let bytes = fs::read(&path).map_err(|_| Error::InvalidAttachmentData)?;
Ok((bytes, filename))
}
}
}
}
impl<P> From<P> for AttachmentSource
where
P: Into<PathBuf>,
{
fn from(value: P) -> Self {
Self::File(value.into())
}
}