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
use super::ArgumentConvert;
use crate::{model::prelude::*, prelude::*};
#[non_exhaustive]
#[derive(Copy, Clone, Debug, Eq, Hash, PartialEq)]
pub enum EmojiParseError {
NotFoundOrMalformed,
}
impl std::error::Error for EmojiParseError {}
impl std::fmt::Display for EmojiParseError {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
match self {
Self::NotFoundOrMalformed => f.write_str("Emoji not found or unknown format"),
}
}
}
#[cfg(feature = "cache")]
#[async_trait::async_trait]
impl ArgumentConvert for Emoji {
type Err = EmojiParseError;
async fn convert(
ctx: &Context,
_guild_id: Option<GuildId>,
_channel_id: Option<ChannelId>,
s: &str,
) -> Result<Self, Self::Err> {
let guilds = ctx.cache.guilds.read().await;
let direct_id = s.parse::<u64>().ok().map(EmojiId);
let id_from_mention = crate::utils::parse_emoji(s).map(|e| e.id);
if let Some(emoji_id) = direct_id.or(id_from_mention) {
if let Some(emoji) = guilds.values().find_map(|guild| guild.emojis.get(&emoji_id)) {
return Ok(emoji.clone());
}
}
if let Some(emoji) = guilds
.values()
.flat_map(|guild| guild.emojis.values())
.find(|emoji| emoji.name.eq_ignore_ascii_case(s))
{
return Ok(emoji.clone());
}
Err(EmojiParseError::NotFoundOrMalformed)
}
}