1//! Types for extensible audio message events ([MSC3927]).
2//!
3//! [MSC3927]: https://github.com/matrix-org/matrix-spec-proposals/pull/3927
45use std::time::Duration;
67use js_int::UInt;
8use ruma_macros::EventContent;
9use serde::{Deserialize, Serialize};
1011#[cfg(feature = "unstable-msc3246")]
12mod amplitude_serde;
1314use super::{
15 file::{CaptionContentBlock, FileContentBlock},
16 message::TextContentBlock,
17 room::message::Relation,
18};
1920/// The payload for an extensible audio message.
21///
22/// This is the new primary type introduced in [MSC3927] and should only be sent in rooms with a
23/// version that supports it. See the documentation of the [`message`] module for more information.
24///
25/// [MSC3927]: https://github.com/matrix-org/matrix-spec-proposals/pull/3927
26/// [`message`]: super::message
27#[derive(Clone, Debug, Serialize, Deserialize, EventContent)]
28#[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
29#[ruma_event(type = "org.matrix.msc1767.audio", kind = MessageLike, without_relation)]
30pub struct AudioEventContent {
31/// The text representations of the message.
32#[serde(rename = "org.matrix.msc1767.text")]
33pub text: TextContentBlock,
3435/// The file content of the message.
36#[serde(rename = "org.matrix.msc1767.file")]
37pub file: FileContentBlock,
3839/// The audio details of the message, if any.
40#[serde(rename = "org.matrix.msc1767.audio_details", skip_serializing_if = "Option::is_none")]
41pub audio_details: Option<AudioDetailsContentBlock>,
4243/// The caption of the message, if any.
44#[serde(rename = "org.matrix.msc1767.caption", skip_serializing_if = "Option::is_none")]
45pub caption: Option<CaptionContentBlock>,
4647/// Whether this message is automated.
48#[cfg(feature = "unstable-msc3955")]
49 #[serde(
50 default,
51 skip_serializing_if = "ruma_common::serde::is_default",
52 rename = "org.matrix.msc1767.automated"
53)]
54pub automated: bool,
5556/// Information about related messages.
57#[serde(
58 flatten,
59 skip_serializing_if = "Option::is_none",
60 deserialize_with = "crate::room::message::relation_serde::deserialize_relation"
61)]
62pub relates_to: Option<Relation<AudioEventContentWithoutRelation>>,
63}
6465impl AudioEventContent {
66/// Creates a new `AudioEventContent` with the given text fallback and file.
67pub fn new(text: TextContentBlock, file: FileContentBlock) -> Self {
68Self {
69 text,
70 file,
71 audio_details: None,
72 caption: None,
73#[cfg(feature = "unstable-msc3955")]
74automated: false,
75 relates_to: None,
76 }
77 }
7879/// Creates a new `AudioEventContent` with the given plain text fallback representation and
80 /// file.
81pub fn with_plain_text(plain_text: impl Into<String>, file: FileContentBlock) -> Self {
82Self {
83 text: TextContentBlock::plain(plain_text),
84 file,
85 audio_details: None,
86 caption: None,
87#[cfg(feature = "unstable-msc3955")]
88automated: false,
89 relates_to: None,
90 }
91 }
92}
9394/// A block for details of audio content.
95#[derive(Clone, Debug, Serialize, Deserialize)]
96#[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
97pub struct AudioDetailsContentBlock {
98/// The duration of the audio in seconds.
99#[serde(with = "ruma_common::serde::duration::secs")]
100pub duration: Duration,
101102/// The waveform representation of the audio content, if any.
103 ///
104 /// This is optional and defaults to an empty array.
105#[cfg(feature = "unstable-msc3246")]
106 #[serde(
107 rename = "org.matrix.msc3246.waveform",
108 default,
109 skip_serializing_if = "Vec::is_empty"
110)]
111pub waveform: Vec<Amplitude>,
112}
113114impl AudioDetailsContentBlock {
115/// Creates a new `AudioDetailsContentBlock` with the given duration.
116pub fn new(duration: Duration) -> Self {
117Self {
118 duration,
119#[cfg(feature = "unstable-msc3246")]
120waveform: Default::default(),
121 }
122 }
123}
124125/// The amplitude of a waveform sample.
126///
127/// Must be an integer between 0 and 256.
128#[cfg(feature = "unstable-msc3246")]
129#[derive(Clone, Copy, Debug, Default, Hash, PartialEq, Eq, PartialOrd, Ord, Serialize)]
130pub struct Amplitude(UInt);
131132#[cfg(feature = "unstable-msc3246")]
133impl Amplitude {
134/// The smallest value that can be represented by this type, 0.
135pub const MIN: u16 = 0;
136137/// The largest value that can be represented by this type, 256.
138pub const MAX: u16 = 256;
139140/// Creates a new `Amplitude` with the given value.
141 ///
142 /// It will saturate if it is bigger than [`Amplitude::MAX`].
143pub fn new(value: u16) -> Self {
144Self(value.min(Self::MAX).into())
145 }
146147/// The value of this `Amplitude`.
148pub fn get(&self) -> UInt {
149self.0
150}
151}
152153#[cfg(feature = "unstable-msc3246")]
154impl From<u16> for Amplitude {
155fn from(value: u16) -> Self {
156Self::new(value)
157 }
158}