use derive_builder::Builder;
use derive_getters::Getters;
use matrix_sdk_base::ruma::events::{
macros::EventContent,
room::message::{
AudioMessageEventContent, FileMessageEventContent, ImageMessageEventContent,
LocationMessageEventContent, MessageType, VideoMessageEventContent,
},
};
use serde::{Deserialize, Serialize};
use super::{BelongsTo, Update};
use crate::Result;
#[derive(Clone, Debug, Deserialize, Serialize)]
#[serde(untagged)]
pub enum FallbackAttachmentContent {
Image(ImageMessageEventContent),
Video(VideoMessageEventContent),
Audio(AudioMessageEventContent),
File(FileMessageEventContent),
Location(LocationMessageEventContent),
}
impl FallbackAttachmentContent {
pub fn name(&self) -> Option<String> {
match self {
FallbackAttachmentContent::Image(ImageMessageEventContent { filename, .. }) => {
filename.clone()
}
FallbackAttachmentContent::Video(VideoMessageEventContent { filename, .. }) => {
filename.clone()
}
FallbackAttachmentContent::Audio(AudioMessageEventContent { filename, .. }) => {
filename.clone()
}
FallbackAttachmentContent::File(FileMessageEventContent { filename, .. }) => {
filename.clone()
}
FallbackAttachmentContent::Location(LocationMessageEventContent { body, .. }) => {
Some(body.clone())
}
}
}
pub fn type_str(&self) -> String {
match self {
FallbackAttachmentContent::Image(_) => "image".to_owned(),
FallbackAttachmentContent::Video(_) => "video".to_owned(),
FallbackAttachmentContent::Audio(_) => "audio".to_owned(),
FallbackAttachmentContent::File(_) => "file".to_owned(),
FallbackAttachmentContent::Location(_) => "location".to_owned(),
}
}
pub fn image(&self) -> Option<ImageMessageEventContent> {
match self {
FallbackAttachmentContent::Image(content) => Some(content.clone()),
_ => None,
}
}
pub fn video(&self) -> Option<VideoMessageEventContent> {
match self {
FallbackAttachmentContent::Video(content) => Some(content.clone()),
_ => None,
}
}
pub fn audio(&self) -> Option<AudioMessageEventContent> {
match self {
FallbackAttachmentContent::Audio(content) => Some(content.clone()),
_ => None,
}
}
pub fn file(&self) -> Option<FileMessageEventContent> {
match self {
FallbackAttachmentContent::File(content) => Some(content.clone()),
_ => None,
}
}
pub fn location(&self) -> Option<LocationMessageEventContent> {
match self {
FallbackAttachmentContent::Location(content) => Some(content.clone()),
_ => None,
}
}
}
#[derive(Clone, Debug, Deserialize, Serialize)]
pub struct LinkAttachmentContent {
#[serde(default, skip_serializing_if = "Option::is_none")]
pub name: Option<String>,
pub link: String,
}
#[derive(Clone, Debug, Deserialize, Serialize)]
#[serde(tag = "type")]
pub enum AttachmentContent {
Image(ImageMessageEventContent),
Video(VideoMessageEventContent),
Audio(AudioMessageEventContent),
File(FileMessageEventContent),
Location(LocationMessageEventContent),
Link(LinkAttachmentContent),
#[serde(untagged)]
Fallback(FallbackAttachmentContent),
}
impl TryFrom<MessageType> for AttachmentContent {
type Error = ();
fn try_from(value: MessageType) -> std::prelude::v1::Result<Self, Self::Error> {
Ok(match value {
MessageType::Image(content) => AttachmentContent::Image(content),
MessageType::Video(content) => AttachmentContent::Video(content),
MessageType::Audio(content) => AttachmentContent::Audio(content),
MessageType::File(content) => AttachmentContent::File(content),
MessageType::Location(content) => AttachmentContent::Location(content),
_ => return Err(()),
})
}
}
impl AttachmentContent {
pub fn name(&self) -> Option<String> {
match self {
AttachmentContent::Image(ImageMessageEventContent { filename, .. }) => filename.clone(),
AttachmentContent::Video(VideoMessageEventContent { filename, .. }) => filename.clone(),
AttachmentContent::Audio(AudioMessageEventContent { filename, .. }) => filename.clone(),
AttachmentContent::File(FileMessageEventContent { filename, .. }) => filename.clone(),
AttachmentContent::Location(LocationMessageEventContent { body, .. }) => {
Some(body.clone())
}
AttachmentContent::Link(LinkAttachmentContent { name, .. }) => name.clone(),
AttachmentContent::Fallback(f) => f.name(),
}
}
pub fn type_str(&self) -> String {
match self {
AttachmentContent::Image(_) => "image".to_owned(),
AttachmentContent::Video(_) => "video".to_owned(),
AttachmentContent::Audio(_) => "audio".to_owned(),
AttachmentContent::File(_) => "file".to_owned(),
AttachmentContent::Location(_) => "location".to_owned(),
AttachmentContent::Link(_) => "link".to_owned(),
AttachmentContent::Fallback(f) => f.type_str(),
}
}
pub fn image(&self) -> Option<ImageMessageEventContent> {
match self {
AttachmentContent::Image(content) => Some(content.clone()),
AttachmentContent::Fallback(f) => f.image(),
_ => None,
}
}
pub fn video(&self) -> Option<VideoMessageEventContent> {
match self {
AttachmentContent::Video(content) => Some(content.clone()),
AttachmentContent::Fallback(f) => f.video(),
_ => None,
}
}
pub fn audio(&self) -> Option<AudioMessageEventContent> {
match self {
AttachmentContent::Audio(content) => Some(content.clone()),
AttachmentContent::Fallback(f) => f.audio(),
_ => None,
}
}
pub fn file(&self) -> Option<FileMessageEventContent> {
match self {
AttachmentContent::File(content) => Some(content.clone()),
AttachmentContent::Fallback(f) => f.file(),
_ => None,
}
}
pub fn location(&self) -> Option<LocationMessageEventContent> {
match self {
AttachmentContent::Location(content) => Some(content.clone()),
AttachmentContent::Fallback(f) => f.location(),
_ => None,
}
}
pub fn link(&self) -> Option<String> {
if let AttachmentContent::Link(LinkAttachmentContent { link, .. }) = self {
Some(link.clone())
} else {
None
}
}
}
#[derive(Clone, Debug, Deserialize, Serialize, EventContent, Builder, Getters)]
#[ruma_event(type = "global.acter.dev.attachment", kind = MessageLike)]
#[builder(name = "AttachmentBuilder", derive(Debug))]
pub struct AttachmentEventContent {
#[builder(setter(into))]
#[serde(rename = "m.relates_to")]
pub on: BelongsTo,
pub content: AttachmentContent,
}
#[derive(Clone, Debug, Deserialize, Serialize, EventContent, Builder)]
#[ruma_event(type = "global.acter.dev.attachment.update", kind = MessageLike)]
#[builder(name = "AttachmentUpdateBuilder", derive(Debug))]
pub struct AttachmentUpdateEventContent {
#[builder(setter(into))]
#[serde(rename = "m.relates_to")]
pub attachment: Update,
pub content: AttachmentContent,
}
impl AttachmentUpdateEventContent {
pub fn apply(&self, task: &mut AttachmentEventContent) -> Result<bool> {
task.content = self.content.clone();
Ok(true)
}
}