1//! Types for extensible video message events ([MSC3553]).
2//!
3//! [MSC3553]: https://github.com/matrix-org/matrix-spec-proposals/pull/3553
45use std::time::Duration;
67use js_int::UInt;
8use ruma_macros::EventContent;
9use serde::{Deserialize, Serialize};
1011use super::{
12 file::{CaptionContentBlock, FileContentBlock},
13 image::ThumbnailContentBlock,
14 message::TextContentBlock,
15 room::message::Relation,
16};
1718/// The payload for an extensible video message.
19///
20/// This is the new primary type introduced in [MSC3553] and should only be sent in rooms with a
21/// version that supports it. See the documentation of the [`message`] module for more information.
22///
23/// [MSC3553]: https://github.com/matrix-org/matrix-spec-proposals/pull/3553
24/// [`message`]: super::message
25#[derive(Clone, Debug, Serialize, Deserialize, EventContent)]
26#[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
27#[ruma_event(type = "org.matrix.msc1767.video", kind = MessageLike, without_relation)]
28pub struct VideoEventContent {
29/// The text representation of the message.
30#[serde(rename = "org.matrix.msc1767.text")]
31pub text: TextContentBlock,
3233/// The file content of the message.
34#[serde(rename = "org.matrix.msc1767.file")]
35pub file: FileContentBlock,
3637/// The video details of the message, if any.
38#[serde(rename = "org.matrix.msc1767.video_details", skip_serializing_if = "Option::is_none")]
39pub video_details: Option<VideoDetailsContentBlock>,
4041/// The thumbnails of the message, if any.
42 ///
43 /// This is optional and defaults to an empty array.
44#[serde(
45 rename = "org.matrix.msc1767.thumbnail",
46 default,
47 skip_serializing_if = "ThumbnailContentBlock::is_empty"
48)]
49pub thumbnail: ThumbnailContentBlock,
5051/// The caption of the message, if any.
52#[serde(rename = "org.matrix.msc1767.caption", skip_serializing_if = "Option::is_none")]
53pub caption: Option<CaptionContentBlock>,
5455/// Whether this message is automated.
56#[cfg(feature = "unstable-msc3955")]
57 #[serde(
58 default,
59 skip_serializing_if = "ruma_common::serde::is_default",
60 rename = "org.matrix.msc1767.automated"
61)]
62pub automated: bool,
6364/// Information about related messages.
65#[serde(
66 flatten,
67 skip_serializing_if = "Option::is_none",
68 deserialize_with = "crate::room::message::relation_serde::deserialize_relation"
69)]
70pub relates_to: Option<Relation<VideoEventContentWithoutRelation>>,
71}
7273impl VideoEventContent {
74/// Creates a new `VideoEventContent` with the given fallback representation and file.
75pub fn new(text: TextContentBlock, file: FileContentBlock) -> Self {
76Self {
77 text,
78 file,
79 video_details: None,
80 thumbnail: Default::default(),
81 caption: None,
82#[cfg(feature = "unstable-msc3955")]
83automated: false,
84 relates_to: None,
85 }
86 }
8788/// Creates a new `VideoEventContent` with the given plain text fallback representation and
89 /// file.
90pub fn with_plain_text(plain_text: impl Into<String>, file: FileContentBlock) -> Self {
91Self {
92 text: TextContentBlock::plain(plain_text),
93 file,
94 video_details: None,
95 thumbnail: Default::default(),
96 caption: None,
97#[cfg(feature = "unstable-msc3955")]
98automated: false,
99 relates_to: None,
100 }
101 }
102}
103104/// A block for details of video content.
105#[derive(Clone, Debug, Serialize, Deserialize)]
106#[cfg_attr(not(ruma_unstable_exhaustive_types), non_exhaustive)]
107pub struct VideoDetailsContentBlock {
108/// The width of the video in pixels.
109pub width: UInt,
110111/// The height of the video in pixels.
112pub height: UInt,
113114/// The duration of the video in seconds.
115#[serde(
116 with = "ruma_common::serde::duration::opt_secs",
117 default,
118 skip_serializing_if = "Option::is_none"
119)]
120pub duration: Option<Duration>,
121}
122123impl VideoDetailsContentBlock {
124/// Creates a new `VideoDetailsContentBlock` with the given height and width.
125pub fn new(width: UInt, height: UInt) -> Self {
126Self { width, height, duration: None }
127 }
128}