Refactor stickers

This commit is contained in:
Maybe Waffle 2022-09-25 20:02:08 +04:00
parent c544dae94c
commit 26ec8b0c7b
3 changed files with 472 additions and 69 deletions

View file

@ -4,11 +4,11 @@ use serde::{Deserialize, Serialize};
/// default. /// default.
/// ///
/// [The official docs](https://core.telegram.org/bots/api#maskposition). /// [The official docs](https://core.telegram.org/bots/api#maskposition).
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)] #[derive(Copy, Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct MaskPosition { pub struct MaskPosition {
/// The part of the face relative to which the mask should be placed. One /// The part of the face relative to which the mask should be placed. One
/// of `forehead`, `eyes`, `mouth`, or `chin`. /// of `forehead`, `eyes`, `mouth`, or `chin`.
pub point: String, pub point: MaskPoint,
/// Shift by X-axis measured in widths of the mask scaled to the face size, /// Shift by X-axis measured in widths of the mask scaled to the face size,
/// from left to right. For example, choosing `-1.0` will place mask just /// from left to right. For example, choosing `-1.0` will place mask just
@ -24,41 +24,45 @@ pub struct MaskPosition {
pub scale: f64, pub scale: f64,
} }
/// The part of the face relative to which the mask should be placed.
#[derive(Copy, Clone, Debug, PartialEq, Serialize, Deserialize)]
#[serde(rename_all = "snake_case")]
pub enum MaskPoint {
Forehead,
Eyes,
Mouth,
Chin,
}
impl MaskPosition { impl MaskPosition {
pub fn new<S>(point: S, x_shift: f64, y_shift: f64, scale: f64) -> Self pub const fn new(point: MaskPoint, x_shift: f64, y_shift: f64, scale: f64) -> Self {
where
S: Into<String>,
{
Self { Self {
point: point.into(), point,
x_shift, x_shift,
y_shift, y_shift,
scale, scale,
} }
} }
pub fn point<S>(mut self, val: S) -> Self pub const fn point(mut self, val: MaskPoint) -> Self {
where self.point = val;
S: Into<String>,
{
self.point = val.into();
self self
} }
#[must_use] #[must_use]
pub fn x_shift(mut self, val: f64) -> Self { pub const fn x_shift(mut self, val: f64) -> Self {
self.x_shift = val; self.x_shift = val;
self self
} }
#[must_use] #[must_use]
pub fn y_shift(mut self, val: f64) -> Self { pub const fn y_shift(mut self, val: f64) -> Self {
self.y_shift = val; self.y_shift = val;
self self
} }
#[must_use] #[must_use]
pub fn scale(mut self, val: f64) -> Self { pub const fn scale(mut self, val: f64) -> Self {
self.scale = val; self.scale = val;
self self
} }

View file

@ -10,25 +10,40 @@ use crate::types::{FileMeta, MaskPosition, PhotoSize};
#[serde_with_macros::skip_serializing_none] #[serde_with_macros::skip_serializing_none]
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)] #[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct Sticker { pub struct Sticker {
/// Identifier for this file. /// Metadata of the sticker file.
pub file_id: String, #[serde(flatten)]
pub file: FileMeta,
/// Unique identifier for this file, which is supposed to be the same over /// Sticker width, in pixels.
/// time and for different bots. Can't be used to download or reuse the ///
/// file. /// You can assume that `max(width, height) = 512`, `min(width, height) <=
pub file_unique_id: String, /// 512`. In other words one dimension is exactly 512 pixels and the other
/// is at most 512 pixels.
/// Sticker width.
pub width: u16, pub width: u16,
/// Sticker height. /// Sticker height, in pixels.
///
/// You can assume that `max(width, height) = 512`, `min(width, height) <=
/// 512`. In other words one dimension is exactly 512 pixels and the other
/// is at most 512 pixels.
pub height: u16, pub height: u16,
/// Kind of this sticker - webp, animated or video. /// Kind of this sticker - regular, mask or custom emoji.
///
/// In other words this represent how the sticker is presented, as a big
/// picture/video, as a mask while editing pictures or as a custom emoji in
/// messages.
#[serde(flatten)] #[serde(flatten)]
pub kind: StickerKind, pub kind: StickerKind,
/// Sticker thumbnail in the .webp or .jpg format. /// Format of this sticker - raster/`.webp`, animated/`.tgs` or
/// video/`.webm`.
///
/// In other words this represents how the sticker is encoded.
#[serde(flatten)]
pub format: StickerFormat,
/// Sticker thumbnail in the `.webp` or `.jpg` format.
pub thumb: Option<PhotoSize>, pub thumb: Option<PhotoSize>,
/// Emoji associated with the sticker. /// Emoji associated with the sticker.
@ -36,29 +51,59 @@ pub struct Sticker {
/// Name of the sticker set to which the sticker belongs. /// Name of the sticker set to which the sticker belongs.
pub set_name: Option<String>, pub set_name: Option<String>,
/// Premium animation for the sticker, if the sticker is premium.
pub premium_animation: Option<FileMeta>,
/// For mask stickers, the position where the mask should be placed.
pub mask_position: Option<MaskPosition>,
/// File size in bytes.
#[serde(default = "crate::types::file::file_size_fallback")]
pub file_size: u32,
} }
/// Kind of a sticker - webp, animated or video. /// Kind of a [`Sticker`] - regular, mask or custom emoji.
///
/// Dataful version of [`StickerType`].
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)] #[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
#[serde(try_from = "StickerKindRaw", into = "StickerKindRaw")] #[serde(tag = "type")]
#[serde(rename_all = "snake_case")]
pub enum StickerKind { pub enum StickerKind {
/// "Normal", raster sticker. /// "Normal", raster, animated or video sticker.
Webp, Regular {
/// [Animated] sticker. /// Premium animation for the sticker, if the sticker is premium.
premium_animation: Option<FileMeta>,
},
/// Mask sticker.
Mask {
/// For mask stickers, the position where the mask should be placed.
mask_position: MaskPosition,
},
/// Custom emoji sticker.
CustomEmoji {
/// A unique identifier of the custom emoji.
// FIXME(waffle): newtype
custom_emoji_id: String,
},
}
/// Type of a [`Sticker`] - regular, mask or custom emoji.
///
/// Dataless version of [`StickerType`].
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
#[serde(tag = "sticker_type")]
#[serde(rename_all = "snake_case")]
pub enum StickerType {
/// "Normal", raster, animated or video sticker.
Regular,
/// Mask sticker.
Mask,
/// Custom emoji sticker.
CustomEmoji,
}
/// Format of a [`Sticker`] - regular/webp, animated/tgs or video/webm.
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
#[serde(try_from = "StickerFormatRaw", into = "StickerFormatRaw")]
pub enum StickerFormat {
/// "Normal", raster, `.webp` sticker.
Raster,
/// [Animated], `.tgs` sticker.
/// ///
/// [Animated]: https://telegram.org/blog/animated-stickers /// [Animated]: https://telegram.org/blog/animated-stickers
Animated, Animated,
/// [Video] sticker. /// [Video], `.webm` sticker.
/// ///
/// [Video]: https://telegram.org/blog/video-stickers-better-reactions /// [Video]: https://telegram.org/blog/video-stickers-better-reactions
Video, Video,
@ -71,8 +116,11 @@ pub enum StickerKind {
/// ///
/// let sticker: Sticker = todo!(); /// let sticker: Sticker = todo!();
/// ///
/// let _ = sticker.is_video(); /// let _ = sticker.is_regular();
/// let _ = sticker.kind.is_video(); /// let _ = sticker.kind.is_regular();
///
/// let _ sticker.mask_position();
/// let _ sticker.kind.mask_position();
/// ``` /// ```
impl Deref for Sticker { impl Deref for Sticker {
type Target = StickerKind; type Target = StickerKind;
@ -82,24 +130,149 @@ impl Deref for Sticker {
} }
} }
impl StickerKind { impl Sticker {
/// Returns `true` is this is a "normal" raster sticker. /// Returns `true` is this is a "normal" raster sticker.
///
/// Alias to [`self.format.is_raster()`].
///
/// [`self.format.is_raster()`]: StickerFormat::is_raster
#[must_use] #[must_use]
pub fn is_webp(&self) -> bool { pub fn is_raster(&self) -> bool {
matches!(self, Self::Webp) self.format.is_raster()
} }
/// Returns `true` is this is an [animated] sticker. /// Returns `true` is this is an [animated] sticker.
/// ///
/// Alias to [`self.format.is_animated()`].
///
/// [`self.format.is_animated()`]: StickerFormat::is_animated
/// [animated]: https://telegram.org/blog/animated-stickers /// [animated]: https://telegram.org/blog/animated-stickers
#[must_use] #[must_use]
pub fn is_animated(&self) -> bool {
self.format.is_animated()
}
/// Returns `true` is this is a [video] sticker.
///
/// Alias to [`self.format.is_video()`].
///
/// [`self.format.is_video()`]: StickerFormat::is_video
/// [video]: https://telegram.org/blog/video-stickers-better-reactions
#[must_use]
pub fn is_video(&self) -> bool {
self.format.is_video()
}
}
impl StickerKind {
/// Converts [`StickerKind`] to [`StickerType`]
#[must_use]
pub fn type_(&self) -> StickerType {
match self {
StickerKind::Regular { .. } => StickerType::Regular,
StickerKind::Mask { .. } => StickerType::Mask,
StickerKind::CustomEmoji { .. } => StickerType::CustomEmoji,
}
}
/// Returns `true` if the sticker kind is [`Regular`].
///
/// [`Regular`]: StickerKind::Regular
#[must_use]
pub fn is_regular(&self) -> bool {
self.type_().is_regular()
}
/// Returns `true` if the sticker kind is [`Mask`].
///
/// [`Mask`]: StickerKind::Mask
#[must_use]
pub fn is_mask(&self) -> bool {
self.type_().is_mask()
}
/// Returns `true` if the sticker kind is [`CustomEmoji`].
///
/// [`CustomEmoji`]: StickerKind::CustomEmoji
#[must_use]
pub fn is_custom_emoji(&self) -> bool {
self.type_().is_custom_emoji()
}
/// Getter for [`StickerKind::Regular::premium_animation`].
pub fn premium_animation(&self) -> Option<&FileMeta> {
if let Self::Regular { premium_animation } = self {
premium_animation.as_ref()
} else {
None
}
}
/// Getter for [`StickerKind::Mask::mask_position`].
pub fn mask_position(&self) -> Option<MaskPosition> {
if let Self::Mask { mask_position } = self {
Some(*mask_position)
} else {
None
}
}
/// Getter for [`StickerKind::CustomEmoji::custom_emoji_id`].
pub fn custom_emoji_id(&self) -> Option<&str> {
if let Self::CustomEmoji { custom_emoji_id } = self {
Some(custom_emoji_id)
} else {
None
}
}
}
impl StickerType {
/// Returns `true` if the sticker type is [`Regular`].
///
/// [`Regular`]: StickerType::Regular
#[must_use]
pub fn is_regular(&self) -> bool {
matches!(self, Self::Regular)
}
/// Returns `true` if the sticker type is [`Mask`].
///
/// [`Mask`]: StickerType::Mask
#[must_use]
pub fn is_mask(&self) -> bool {
matches!(self, Self::Mask)
}
/// Returns `true` if the sticker type is [`CustomEmoji`].
///
/// [`CustomEmoji`]: StickerType::CustomEmoji
#[must_use]
pub fn is_custom_emoji(&self) -> bool {
matches!(self, Self::CustomEmoji)
}
}
impl StickerFormat {
/// Returns `true` if the sticker format is [`Raster`].
///
/// [`Raster`]: StickerFormat::Raster
#[must_use]
pub fn is_raster(&self) -> bool {
matches!(self, Self::Raster)
}
/// Returns `true` if the sticker format is [`Animated`].
///
/// [`Animated`]: StickerFormat::Animated
#[must_use]
pub fn is_animated(&self) -> bool { pub fn is_animated(&self) -> bool {
matches!(self, Self::Animated) matches!(self, Self::Animated)
} }
/// Returns `true` is this is a [video] sticker. /// Returns `true` if the sticker format is [`Video`].
/// ///
/// [video]: https://telegram.org/blog/video-stickers-better-reactions /// [`Video`]: StickerFormat::Video
#[must_use] #[must_use]
pub fn is_video(&self) -> bool { pub fn is_video(&self) -> bool {
matches!(self, Self::Video) matches!(self, Self::Video)
@ -107,22 +280,22 @@ impl StickerKind {
} }
#[derive(Serialize, Deserialize)] #[derive(Serialize, Deserialize)]
struct StickerKindRaw { struct StickerFormatRaw {
is_animated: bool, is_animated: bool,
is_video: bool, is_video: bool,
} }
impl TryFrom<StickerKindRaw> for StickerKind { impl TryFrom<StickerFormatRaw> for StickerFormat {
type Error = &'static str; type Error = &'static str;
fn try_from( fn try_from(
StickerKindRaw { StickerFormatRaw {
is_animated, is_animated,
is_video, is_video,
}: StickerKindRaw, }: StickerFormatRaw,
) -> Result<Self, Self::Error> { ) -> Result<Self, Self::Error> {
let ret = match (is_animated, is_video) { let ret = match (is_animated, is_video) {
(false, false) => Self::Webp, (false, false) => Self::Raster,
(true, false) => Self::Animated, (true, false) => Self::Animated,
(false, true) => Self::Video, (false, true) => Self::Video,
(true, true) => return Err("`is_animated` and `is_video` present at the same time"), (true, true) => return Err("`is_animated` and `is_video` present at the same time"),
@ -132,21 +305,146 @@ impl TryFrom<StickerKindRaw> for StickerKind {
} }
} }
impl From<StickerKind> for StickerKindRaw { impl From<StickerFormat> for StickerFormatRaw {
fn from(kind: StickerKind) -> Self { fn from(kind: StickerFormat) -> Self {
match kind { match kind {
StickerKind::Webp => Self { StickerFormat::Raster => Self {
is_animated: false, is_animated: false,
is_video: false, is_video: false,
}, },
StickerKind::Animated => Self { StickerFormat::Animated => Self {
is_animated: true, is_animated: true,
is_video: false, is_video: false,
}, },
StickerKind::Video => Self { StickerFormat::Video => Self {
is_animated: false, is_animated: false,
is_video: true, is_video: true,
}, },
} }
} }
} }
#[cfg(test)]
mod tests {
use crate::types::{MaskPoint, Sticker, StickerFormat, StickerType};
#[test]
fn mask_serde() {
// Taken from a real (mask) sticker set
let json = r#"{
"width": 512,
"height": 512,
"emoji": "🎭",
"set_name": "Coronamask",
"is_animated": false,
"is_video": false,
"type": "mask",
"mask_position": {
"point": "forehead",
"x_shift": -0.0125,
"y_shift": 0.5525,
"scale": 1.94
},
"thumb": {
"file_id": "AAMCAQADFQABYzA0qlYHijpjMzMwBFKnEVE5XdkAAjIKAAK_jJAE1TRw7D936M8BAAdtAAMpBA",
"file_unique_id": "AQADMgoAAr-MkARy",
"file_size": 11028,
"width": 320,
"height": 320
},
"file_id": "CAACAgEAAxUAAWMwNKpWB4o6YzMzMARSpxFROV3ZAAIyCgACv4yQBNU0cOw_d-jPKQQ",
"file_unique_id": "AgADMgoAAr-MkAQ",
"file_size": 18290
}"#;
let sticker: Sticker = serde_json::from_str(json).unwrap();
// Assert some basic properties are correctly deserialized
assert_eq!(sticker.type_(), StickerType::Mask);
assert_eq!(sticker.mask_position().unwrap().point, MaskPoint::Forehead);
assert_eq!(sticker.is_animated(), false);
assert_eq!(sticker.is_video(), false);
assert_eq!(sticker.thumb.clone().unwrap().file_size, 11028);
assert_eq!(sticker.file.file_size, 18290);
assert_eq!(sticker.width, 512);
assert_eq!(sticker.height, 512);
let json2 = serde_json::to_string(&sticker).unwrap();
let sticker2: Sticker = serde_json::from_str(&json2).unwrap();
assert_eq!(sticker, sticker2);
}
#[test]
fn regular_serde() {
// Taken from a real sticker set
let json = r#"{
"width": 463,
"height": 512,
"emoji": "🍿",
"set_name": "menhera2",
"is_animated": false,
"is_video": false,
"type": "regular",
"thumb": {
"file_id": "AAMCAgADFQABYzBxOJ1GWrttqL7FSRwdAtrq-AkAAtkHAALBGJ4LUUUh5CUew90BAAdtAAMpBA",
"file_unique_id": "AQAD2QcAAsEYngty",
"file_size": 4558,
"width": 116,
"height": 128
},
"file_id": "CAACAgIAAxUAAWMwcTidRlq7bai-xUkcHQLa6vgJAALZBwACwRieC1FFIeQlHsPdKQQ",
"file_unique_id": "AgAD2QcAAsEYngs",
"file_size": 25734
}"#;
let sticker: Sticker = serde_json::from_str(json).unwrap();
// Assert some basic properties are correctly deserialized
assert_eq!(sticker.type_(), StickerType::Regular);
assert_eq!(sticker.premium_animation(), None);
assert_eq!(sticker.is_animated(), false);
assert_eq!(sticker.is_video(), false);
assert_eq!(sticker.thumb.clone().unwrap().file_size, 4558);
assert_eq!(sticker.file.file_size, 25734);
assert_eq!(sticker.width, 463);
assert_eq!(sticker.height, 512);
assert_eq!(sticker.set_name.as_deref(), Some("menhera2"));
let json2 = serde_json::to_string(&sticker).unwrap();
let sticker2: Sticker = serde_json::from_str(&json2).unwrap();
assert_eq!(sticker, sticker2);
}
#[test]
fn sticker_format_serde() {
{
let json = r#"{"is_animation":false,"is_video":false}"#;
let fmt: StickerFormat = serde_json::from_str(json).unwrap();
assert_eq!(fmt, StickerFormat::Raster);
let json2 = serde_json::to_string(&fmt).unwrap();
assert_eq!(json, json2);
}
{
let json = r#"{"is_animation":true,"is_video":false}"#;
let fmt: StickerFormat = serde_json::from_str(json).unwrap();
assert_eq!(fmt, StickerFormat::Animated);
let json2 = serde_json::to_string(&fmt).unwrap();
assert_eq!(json, json2);
}
{
let json = r#"{"is_animation":false,"is_video":true}"#;
let fmt: StickerFormat = serde_json::from_str(json).unwrap();
assert_eq!(fmt, StickerFormat::Video);
let json2 = serde_json::to_string(&fmt).unwrap();
assert_eq!(json, json2);
}
{
let json = r#"{"is_animation":true,"is_video":true}"#;
let fmt: Result<StickerFormat, _> = serde_json::from_str(json);
assert!(fmt.is_err());
}
}
}

View file

@ -2,7 +2,7 @@ use std::ops::Deref;
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use crate::types::{PhotoSize, Sticker, StickerKind}; use crate::types::{PhotoSize, Sticker, StickerFormat, StickerType};
/// This object represents a sticker set. /// This object represents a sticker set.
/// ///
@ -15,33 +15,134 @@ pub struct StickerSet {
/// Sticker set title. /// Sticker set title.
pub title: String, pub title: String,
/// Sticker kind shared by all stickers in this set. /// Sticker type shared by all stickers in this set.
pub kind: StickerKind, #[serde(flatten)]
pub kind: StickerType,
/// `true`, if the sticker set contains masks. /// Sticker format shared by all stickers in this set.
pub contains_masks: bool, #[serde(flatten)]
pub format: StickerFormat,
/// List of all set stickers. /// List of all set stickers.
pub stickers: Vec<Sticker>, pub stickers: Vec<Sticker>,
/// Sticker set thumbnail in the .WEBP or .TGS format. /// Sticker set thumbnail in the `.webp`, `.tgs` or `.webm` format.
pub thumb: Option<PhotoSize>, pub thumb: Option<PhotoSize>,
} }
/// This allows calling [`StickerKind`]'s methods directly on [`StickerSet`]. /// This allows calling [`StickerType`]'s methods directly on [`StickerSet`].
/// ///
/// ```no_run /// ```no_run
/// use teloxide_core::types::StickerSet; /// use teloxide_core::types::StickerSet;
/// ///
/// let sticker: StickerSet = todo!(); /// let sticker: StickerSet = todo!();
/// ///
/// let _ = sticker.is_video(); /// let _ = sticker.is_mask();
/// let _ = sticker.kind.is_video(); /// let _ = sticker.kind.is_mask();
/// ``` /// ```
impl Deref for StickerSet { impl Deref for StickerSet {
type Target = StickerKind; type Target = StickerType;
fn deref(&self) -> &Self::Target { fn deref(&self) -> &Self::Target {
&self.kind &self.kind
} }
} }
impl StickerSet {
/// Returns `true` is this is a "normal" raster sticker.
///
/// Alias to [`self.format.is_raster()`].
///
/// [`self.format.is_raster()`]: StickerFormat::is_raster
#[must_use]
pub fn is_raster(&self) -> bool {
self.format.is_raster()
}
/// Returns `true` is this is an [animated] sticker.
///
/// Alias to [`self.format.is_animated()`].
///
/// [`self.format.is_animated()`]: StickerFormat::is_animated
/// [animated]: https://telegram.org/blog/animated-stickers
#[must_use]
pub fn is_animated(&self) -> bool {
self.format.is_animated()
}
/// Returns `true` is this is a [video] sticker.
///
/// Alias to [`self.format.is_video()`].
///
/// [`self.format.is_video()`]: StickerFormat::is_video
/// [video]: https://telegram.org/blog/video-stickers-better-reactions
#[must_use]
pub fn is_video(&self) -> bool {
self.format.is_video()
}
}
#[cfg(test)]
mod tests {
use crate::types::StickerSet;
#[test]
fn smoke_serde() {
// https://t.me/addstickers/teloxide_test
let json = r#"{
"name": "teloxide_test",
"title": "teloxide-test",
"is_animated": false,
"is_video": false,
"sticker_type": "regular",
"contains_masks": false,
"stickers": [
{
"width": 512,
"height": 512,
"emoji": "⚙️",
"set_name": "teloxide_test",
"is_animated": false,
"is_video": false,
"type": "regular",
"thumb": {
"file_id": "AAMCAQADFQABYzB4ATH0sqXx351gZ5GpY1Z3Tl8AAlgCAAJ1t4hFbxNCoAg1-akBAAdtAAMpBA",
"file_unique_id": "AQADWAIAAnW3iEVy",
"file_size": 7698,
"width": 320,
"height": 320
},
"file_id": "CAACAgEAAxUAAWMweAEx9LKl8d-dYGeRqWNWd05fAAJYAgACdbeIRW8TQqAINfmpKQQ",
"file_unique_id": "AgADWAIAAnW3iEU",
"file_size": 12266
},
{
"width": 512,
"height": 512,
"emoji": "⚙️",
"set_name": "teloxide_test",
"is_animated": false,
"is_video": false,
"type": "regular",
"thumb": {
"file_id": "AAMCAQADFQABYzB4AcABR8-MuvGagis9Pk6liSAAAs8DAAL2YYBFNbvduoN1p7oBAAdtAAMpBA",
"file_unique_id": "AQADzwMAAvZhgEVy",
"file_size": 7780,
"width": 320,
"height": 320
},
"file_id": "CAACAgEAAxUAAWMweAHAAUfPjLrxmoIrPT5OpYkgAALPAwAC9mGARTW73bqDdae6KQQ",
"file_unique_id": "AgADzwMAAvZhgEU",
"file_size": 12158
}
]
}"#;
let set: StickerSet = serde_json::from_str(json).unwrap();
assert!(set.is_raster());
assert!(set.is_regular());
assert!(set.thumb.is_none());
assert_eq!(set.stickers.len(), 2);
}
}