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
use super::InputFile;
use crate::types::{
file,
parameters::{ParseMode, Text},
};
use serde::ser::SerializeMap;
#[derive(Debug, PartialEq, Eq, Clone, Hash)]
#[must_use]
pub struct Photo {
pub(crate) media: InputFile,
pub(crate) caption: Option<String>,
pub(crate) parse_mode: Option<ParseMode>,
}
impl Photo {
const fn new(media: InputFile) -> Self {
Self {
media,
caption: None,
parse_mode: None,
}
}
pub fn with_bytes(bytes: impl Into<Vec<u8>>) -> Self {
Self::new(InputFile::File {
filename: "photo.jpg".into(),
bytes: bytes.into(),
})
}
pub fn with_id(id: file::Id) -> Self {
assert!(
!id.0.starts_with("attach://"),
"\n[tbot]: Photo's ID cannot start with `attach://`\n",
);
Self::new(InputFile::Id(id))
}
pub fn with_url(url: impl Into<String>) -> Self {
let url = url.into();
assert!(
!url.starts_with("attach://"),
"\n[tbot]: Photo's URL cannot start with `attach://`\n",
);
Self::new(InputFile::Url(url))
}
pub fn caption(mut self, caption: impl Into<Text>) -> Self {
let caption = caption.into();
self.caption = Some(caption.text);
self.parse_mode = caption.parse_mode;
self
}
pub(crate) fn serialize_with_name<S>(
&self,
serializer: S,
name: &str,
) -> Result<S::Ok, S::Error>
where
S: serde::Serializer,
{
let mut map = serializer.serialize_map(None)?;
map.serialize_entry("type", "photo")?;
map.serialize_entry("media", &self.media.with_name(name))?;
if let Some(caption) = &self.caption {
map.serialize_entry("caption", caption)?;
}
if let Some(parse_mode) = self.parse_mode {
map.serialize_entry("parse_mode", &parse_mode)?;
}
map.end()
}
}
impl serde::Serialize for Photo {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: serde::Serializer,
{
self.serialize_with_name(serializer, "photo")
}
}