Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

MSC2530: added the ability to send media with captions #3226

Merged
merged 23 commits into from
Mar 19, 2024
Merged
Show file tree
Hide file tree
Changes from 17 commits
Commits
Show all changes
23 commits
Select commit Hold shift + click to select a range
46a5f0c
ffi: Expose filename and formatted body fields for media captions
SpiritCroc Dec 30, 2023
64aa264
Merge remote-tracking branch 'origin/main'
surakin Mar 11, 2024
705adfe
Merge remote-tracking branch 'origin/main'
surakin Mar 13, 2024
a4301c6
Merge remote-tracking branch 'origin/main'
surakin Mar 14, 2024
ed39ed5
MSC2530: added the ability to send media with captions
surakin Mar 16, 2024
7aa59a6
signoff
surakin Mar 16, 2024
9bdd989
fixing the import messup
surakin Mar 16, 2024
064b7e0
Merge remote-tracking branch 'upstream/main' into msc2530
surakin Mar 16, 2024
0ba1afb
Merge branch 'matrix-org:main' into msc2530
surakin Mar 16, 2024
7a7546e
Merge branch 'msc2530' of github.com:surakin/matrix-rust-sdk into msc…
surakin Mar 16, 2024
e21a0d2
fix missing parameters in documentation
surakin Mar 17, 2024
e165e1c
fix formatting
surakin Mar 17, 2024
74ca821
move optional parameters to the end
surakin Mar 17, 2024
24931fe
more formatting fixes
surakin Mar 17, 2024
4c6fa67
more formatting fixes
surakin Mar 17, 2024
76d58dc
rename url parameter to filename in send_attachment and helpers
surakin Mar 17, 2024
3ab9e1a
fix send_attachment documentation example
surakin Mar 17, 2024
1798315
move caption and formatted_caption into attachmentconfig
surakin Mar 18, 2024
053a656
fix formatting
surakin Mar 18, 2024
79fb2c6
fix formatting
surakin Mar 18, 2024
91816e3
fix formatting (hopefully the last one)
surakin Mar 18, 2024
9f89e2d
updated stale comments
surakin Mar 19, 2024
58529c3
simplify attachment message comments
surakin Mar 19, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
74 changes: 62 additions & 12 deletions bindings/matrix-sdk-ffi/src/timeline/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -38,8 +38,8 @@ use ruma::{
receipt::ReceiptThread,
relation::Annotation,
room::message::{
ForwardThread, LocationMessageEventContent, MessageType,
RoomMessageEventContentWithoutRelation,
FormattedBody as RumaFormattedBody, ForwardThread, LocationMessageEventContent,
MessageType, RoomMessageEventContentWithoutRelation,
},
AnyMessageLikeEventContent,
},
Expand All @@ -52,19 +52,21 @@ use tokio::{
use tracing::{error, info, warn};
use uuid::Uuid;

use self::content::{Reaction, ReactionSenderData, TimelineItemContent};
use crate::{
client::ProgressWatcher,
error::{ClientError, RoomError},
helpers::unwrap_or_clone_arc,
ruma::{AssetType, AudioInfo, FileInfo, ImageInfo, PollKind, ThumbnailInfo, VideoInfo},
ruma::{
AssetType, AudioInfo, FileInfo, FormattedBody, ImageInfo, PollKind, ThumbnailInfo,
VideoInfo,
},
task_handle::TaskHandle,
RUNTIME,
};

mod content;

pub use self::content::{Reaction, ReactionSenderData, TimelineItemContent};

#[derive(uniffi::Object)]
#[repr(transparent)]
pub struct Timeline {
Expand Down Expand Up @@ -106,12 +108,19 @@ impl Timeline {

async fn send_attachment(
&self,
url: String,
filename: String,
mime_type: Mime,
attachment_config: AttachmentConfig,
caption: Option<String>,
formatted: Option<FormattedBody>,
progress_watcher: Option<Box<dyn ProgressWatcher>>,
) -> Result<(), RoomError> {
let request = self.inner.send_attachment(url, mime_type, attachment_config);
let formatted: Option<RumaFormattedBody> = match formatted {
Some(p) => Some(RumaFormattedBody::from(p)),
None => None,
};
surakin marked this conversation as resolved.
Show resolved Hide resolved
let request =
self.inner.send_attachment(filename, mime_type, attachment_config, caption, formatted);
if let Some(progress_watcher) = progress_watcher {
let mut subscriber = request.subscribe_to_send_progress();
RUNTIME.spawn(async move {
Expand Down Expand Up @@ -218,6 +227,8 @@ impl Timeline {
url: String,
thumbnail_url: Option<String>,
image_info: ImageInfo,
caption: Option<String>,
formatted: Option<FormattedBody>,
progress_watcher: Option<Box<dyn ProgressWatcher>>,
) -> Arc<SendAttachmentJoinHandle> {
SendAttachmentJoinHandle::new(RUNTIME.spawn(async move {
Expand All @@ -240,7 +251,15 @@ impl Timeline {
_ => AttachmentConfig::new().info(attachment_info),
};

self.send_attachment(url, mime_type, attachment_config, progress_watcher).await
self.send_attachment(
url,
mime_type,
attachment_config,
caption,
formatted,
progress_watcher,
)
.await
}))
}

Expand All @@ -249,6 +268,8 @@ impl Timeline {
url: String,
thumbnail_url: Option<String>,
video_info: VideoInfo,
caption: Option<String>,
formatted: Option<FormattedBody>,
progress_watcher: Option<Box<dyn ProgressWatcher>>,
) -> Arc<SendAttachmentJoinHandle> {
SendAttachmentJoinHandle::new(RUNTIME.spawn(async move {
Expand All @@ -271,14 +292,24 @@ impl Timeline {
_ => AttachmentConfig::new().info(attachment_info),
};

self.send_attachment(url, mime_type, attachment_config, progress_watcher).await
self.send_attachment(
url,
mime_type,
attachment_config,
caption,
formatted,
progress_watcher,
)
.await
}))
}

pub fn send_audio(
self: Arc<Self>,
url: String,
audio_info: AudioInfo,
caption: Option<String>,
formatted: Option<FormattedBody>,
progress_watcher: Option<Box<dyn ProgressWatcher>>,
) -> Arc<SendAttachmentJoinHandle> {
SendAttachmentJoinHandle::new(RUNTIME.spawn(async move {
Expand All @@ -293,7 +324,15 @@ impl Timeline {
let attachment_info = AttachmentInfo::Audio(base_audio_info);
let attachment_config = AttachmentConfig::new().info(attachment_info);

self.send_attachment(url, mime_type, attachment_config, progress_watcher).await
self.send_attachment(
url,
mime_type,
attachment_config,
caption,
formatted,
progress_watcher,
)
.await
}))
}

Expand All @@ -302,6 +341,8 @@ impl Timeline {
url: String,
audio_info: AudioInfo,
waveform: Vec<u16>,
caption: Option<String>,
formatted: Option<FormattedBody>,
progress_watcher: Option<Box<dyn ProgressWatcher>>,
) -> Arc<SendAttachmentJoinHandle> {
SendAttachmentJoinHandle::new(RUNTIME.spawn(async move {
Expand All @@ -317,7 +358,15 @@ impl Timeline {
AttachmentInfo::Voice { audio_info: base_audio_info, waveform: Some(waveform) };
let attachment_config = AttachmentConfig::new().info(attachment_info);

self.send_attachment(url, mime_type, attachment_config, progress_watcher).await
self.send_attachment(
url,
mime_type,
attachment_config,
caption,
formatted,
progress_watcher,
)
.await
}))
}

Expand All @@ -339,7 +388,8 @@ impl Timeline {
let attachment_info = AttachmentInfo::File(base_file_info);
let attachment_config = AttachmentConfig::new().info(attachment_info);

self.send_attachment(url, mime_type, attachment_config, progress_watcher).await
self.send_attachment(url, mime_type, attachment_config, None, None, progress_watcher)
.await
}))
}

Expand Down
30 changes: 23 additions & 7 deletions crates/matrix-sdk-ui/src/timeline/futures.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,31 +4,38 @@ use eyeball::{SharedObservable, Subscriber};
use matrix_sdk::{attachment::AttachmentConfig, TransmissionProgress};
use matrix_sdk_base::boxed_into_future;
use mime::Mime;
use ruma::events::room::message::FormattedBody;
use tracing::{Instrument as _, Span};

use super::{Error, Timeline};

pub struct SendAttachment<'a> {
timeline: &'a Timeline,
url: String,
filename: String,
mime_type: Mime,
config: AttachmentConfig,
caption: Option<String>,
formatted: Option<FormattedBody>,
tracing_span: Span,
pub(crate) send_progress: SharedObservable<TransmissionProgress>,
}

impl<'a> SendAttachment<'a> {
pub(crate) fn new(
timeline: &'a Timeline,
url: String,
filename: String,
mime_type: Mime,
config: AttachmentConfig,
caption: Option<String>,
formatted: Option<FormattedBody>,
) -> Self {
Self {
timeline,
url,
filename,
mime_type,
config,
caption,
formatted,
tracing_span: Span::current(),
send_progress: Default::default(),
}
Expand All @@ -47,18 +54,27 @@ impl<'a> IntoFuture for SendAttachment<'a> {
boxed_into_future!(extra_bounds: 'a);

fn into_future(self) -> Self::IntoFuture {
let Self { timeline, url, mime_type, config, tracing_span, send_progress } = self;
let Self {
timeline,
filename,
mime_type,
config,
caption,
formatted,
tracing_span,
send_progress,
} = self;
let fut = async move {
let body = Path::new(&url)
let body = Path::new(&filename)
.file_name()
.ok_or(Error::InvalidAttachmentFileName)?
.to_str()
.expect("path was created from UTF-8 string, hence filename part is UTF-8 too");
let data = fs::read(&url).map_err(|_| Error::InvalidAttachmentData)?;
let data = fs::read(&filename).map_err(|_| Error::InvalidAttachmentData)?;

timeline
.room()
.send_attachment(body, &mime_type, data, config)
.send_attachment(body, &mime_type, data, config, caption, formatted)
.with_send_progress_observable(send_progress)
.await
.map_err(|_| Error::FailedSendingAttachment)?;
Expand Down
18 changes: 13 additions & 5 deletions crates/matrix-sdk-ui/src/timeline/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -45,8 +45,9 @@ use ruma::{
relation::Annotation,
room::{
message::{
AddMentions, ForwardThread, OriginalRoomMessageEvent, ReplacementMetadata,
RoomMessageEventContent, RoomMessageEventContentWithoutRelation,
AddMentions, FormattedBody, ForwardThread, OriginalRoomMessageEvent,
ReplacementMetadata, RoomMessageEventContent,
RoomMessageEventContentWithoutRelation,
},
redaction::RoomRedactionEventContent,
},
Expand Down Expand Up @@ -513,21 +514,28 @@ impl Timeline {
///
/// # Arguments
///
/// * `url` - The url for the file to be sent
/// * `filename` - The filename of the file to be sent
///
/// * `mime_type` - The attachment's mime type
///
/// * `config` - An attachment configuration object containing details about
/// the attachment
///
/// * `caption` - An optional caption of this attachment
///
/// * `formatted` - An optional formatted caption of this attachment
///
surakin marked this conversation as resolved.
Show resolved Hide resolved
/// like a thumbnail, its size, duration etc.
#[instrument(skip_all)]
pub fn send_attachment(
&self,
url: String,
filename: String,
mime_type: Mime,
config: AttachmentConfig,
caption: Option<String>,
formatted: Option<FormattedBody>,
) -> SendAttachment<'_> {
SendAttachment::new(self, url, mime_type, config)
SendAttachment::new(self, filename, mime_type, config, caption, formatted)
}

/// Retry sending a message that previously failed to send.
Expand Down
2 changes: 2 additions & 0 deletions crates/matrix-sdk/src/attachment.rs
Original file line number Diff line number Diff line change
Expand Up @@ -332,6 +332,8 @@ impl Default for AttachmentConfig {
/// if let Some(room) = client.get_room(&room_id) {
/// room.send_attachment(
/// "My favorite cat",
/// None,
/// "my_favorite_cat.jpg",
/// &mime::IMAGE_JPEG,
/// image,
/// config,
Expand Down
21 changes: 16 additions & 5 deletions crates/matrix-sdk/src/encryption/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -49,8 +49,8 @@ use ruma::{
assign,
events::room::{
message::{
AudioMessageEventContent, FileInfo, FileMessageEventContent, ImageMessageEventContent,
MessageType, VideoInfo, VideoMessageEventContent,
AudioMessageEventContent, FileInfo, FileMessageEventContent, FormattedBody,
ImageMessageEventContent, MessageType, VideoInfo, VideoMessageEventContent,
},
ImageInfo, MediaSource, ThumbnailInfo,
},
Expand Down Expand Up @@ -298,14 +298,17 @@ impl Client {
}

/// Encrypt and upload the file to be read from `reader` and construct an
/// attachment message with `body`, `content_type`, `info` and `thumbnail`.
/// attachment message with `body`, `content_type`, `info` and `thumbnail`,
/// optionanly with `formatted` and `filename`.
surakin marked this conversation as resolved.
Show resolved Hide resolved
pub(crate) async fn prepare_encrypted_attachment_message(
&self,
body: &str,
filename: &str,
content_type: &mime::Mime,
data: Vec<u8>,
info: Option<AttachmentInfo>,
thumbnail: Option<Thumbnail>,
caption: Option<String>,
formatted: Option<FormattedBody>,
send_progress: SharedObservable<TransmissionProgress>,
) -> Result<MessageType> {
let upload_thumbnail =
Expand All @@ -321,6 +324,12 @@ impl Client {
let ((thumbnail_source, thumbnail_info), file) =
try_join(upload_thumbnail, upload_attachment).await?;

let filename = filename.to_owned();
let (body, filename) = match caption {
Some(caption) => (caption, Some(filename)),
None => (filename, None),
};
surakin marked this conversation as resolved.
Show resolved Hide resolved

Ok(match content_type.type_() {
mime::IMAGE => {
let info = assign!(info.map(ImageInfo::from).unwrap_or_default(), {
Expand All @@ -329,7 +338,9 @@ impl Client {
thumbnail_info
});
let content = assign!(ImageMessageEventContent::encrypted(body.to_owned(), file), {
info: Some(Box::new(info))
info: Some(Box::new(info)),
formatted: formatted,
filename: filename
surakin marked this conversation as resolved.
Show resolved Hide resolved
});
MessageType::Image(content)
}
Expand Down
Loading
Loading