use super::{Formatter as TextFormatter, FormatterErrorKind};
use crate::types::{
MessageEntity, MessageEntityCustomEmoji, MessageEntityDateTime, MessageEntityPre,
MessageEntityTextLink, MessageEntityTextMention,
};
use std::fmt::Display;
const CHARS: [char; 18] = [
'_', '*', '[', ']', '(', ')', '~', '`', '>', '#', '+', '-', '=', '|', '{', '}', '.', '!',
];
#[derive(Debug, Clone)]
pub struct Formatter;
impl Formatter {
#[inline]
#[must_use]
pub const fn new() -> Self {
Self {}
}
}
impl Default for Formatter {
#[inline]
fn default() -> Self {
Self::new()
}
}
impl TextFormatter for Formatter {
fn bold<T>(&self, text: T) -> String
where
T: Display,
{
format!("*{text}*")
}
fn italic<T>(&self, text: T) -> String
where
T: Display,
{
format!("_\r{text}_\r")
}
fn underline<T>(&self, text: T) -> String
where
T: Display,
{
format!("__\r{text}__\r")
}
fn strikethrough<T>(&self, text: T) -> String
where
T: Display,
{
format!("~{text}~")
}
fn spoiler<T>(&self, text: T) -> String
where
T: Display,
{
format!("|{text}|")
}
fn blockquote<T>(&self, text: T) -> String
where
T: Display,
{
text.to_string()
.lines()
.map(|line| format!(">{line}"))
.collect::<Vec<_>>()
.join("\n")
}
fn expandable_blockquote<T>(&self, text: T) -> String
where
T: Display,
{
let mut text = self.blockquote(text);
text.push_str("||");
text
}
fn text_link<T, U>(&self, text: T, url: U) -> String
where
T: Display,
U: Display,
{
format!("[{text}]({url})")
}
fn text_mention<T>(&self, text: T, user_id: i64) -> String
where
T: Display,
{
self.text_link(text, format!("tg://user?id={user_id}"))
}
fn custom_emoji<T, E>(&self, emoji: T, emoji_id: E) -> String
where
T: Display,
E: Display,
{
format!(
"!{}",
self.text_link(emoji, format!("tg://emoji?id={emoji_id}"),)
)
}
fn code<T>(&self, text: T) -> String
where
T: Display,
{
format!("`{text}`")
}
fn pre<T>(&self, text: T) -> String
where
T: Display,
{
format!("```\n{text}\n```")
}
fn pre_language<T, L>(&self, text: T, language: L) -> String
where
T: Display,
L: Display,
{
format!("```{language}\n{text}\n```")
}
fn date_time<T>(&self, text: T, unix_time: i64) -> String
where
T: Display,
{
format!(
"!{}",
self.text_link(text, format!("tg://time?unix={unix_time}"))
)
}
fn date_time_with_format<T, F>(&self, text: T, unix_time: i64, date_time_format: F) -> String
where
T: Display,
F: Display,
{
format!(
"!{}",
self.text_link(
text,
format!("tg://time?unix={unix_time}&format={date_time_format}")
)
)
}
fn quote<T>(&self, text: T) -> String
where
T: Display,
{
let text = text.to_string();
text.chars()
.fold(String::with_capacity(text.len()), |mut string, ch| {
if CHARS.contains(&ch) {
string.push('\\');
}
string.push(ch);
string
})
}
fn apply_entity<T>(&self, text: T, entity: &MessageEntity) -> Result<String, FormatterErrorKind>
where
T: Display,
{
let text = text.to_string();
let text_len = text.len();
if text_len == 0 {
return Err(FormatterErrorKind::EmptyText);
}
let offset = usize::try_from(entity.offset()).unwrap();
let length = usize::try_from(entity.length()).unwrap();
if offset + length > text_len {
return Err(FormatterErrorKind::IndexOutOfBounds);
}
let previous_text = &text[..offset];
let editable_text = &text[offset..offset + length];
let next_text = &text[offset + length..];
let edited_text = match entity {
MessageEntity::Mention(_) => format!("@{editable_text}"),
MessageEntity::Hashtag(_) => format!("#{editable_text}"),
MessageEntity::Cashtag(_) => format!("${editable_text}"),
MessageEntity::BotCommand(_) => format!("/{editable_text}"),
MessageEntity::Url(_) | MessageEntity::Email(_) | MessageEntity::PhoneNumber(_) => {
editable_text.to_owned()
}
MessageEntity::Bold(_) => self.bold(editable_text),
MessageEntity::Italic(_) => self.italic(editable_text),
MessageEntity::Underline(_) => self.underline(editable_text),
MessageEntity::Strikethrough(_) => self.strikethrough(editable_text),
MessageEntity::Spoiler(_) => self.spoiler(editable_text),
MessageEntity::Blockquote(_) => self.blockquote(editable_text),
MessageEntity::ExpandableBlockquote(_) => self.expandable_blockquote(editable_text),
MessageEntity::Code(_) => self.code(editable_text),
MessageEntity::Pre(MessageEntityPre {
language, ..
}) => match language {
Some(language) => self.pre_language(editable_text, language),
None => self.pre(editable_text),
},
MessageEntity::TextLink(MessageEntityTextLink {
url, ..
}) => self.text_link(editable_text, url),
MessageEntity::TextMention(MessageEntityTextMention {
user, ..
}) => self.text_mention(editable_text, user.id),
MessageEntity::CustomEmoji(MessageEntityCustomEmoji {
custom_emoji_id, ..
}) => self.custom_emoji(editable_text, custom_emoji_id),
MessageEntity::DateTime(MessageEntityDateTime {
unix_time,
date_time_format,
..
}) => match date_time_format {
Some(date_time_format) => {
self.date_time_with_format(editable_text, *unix_time, date_time_format)
}
None => self.date_time(editable_text, *unix_time),
},
};
Ok(format!("{previous_text}{edited_text}{next_text}"))
}
}
pub const FORMATTER: Formatter = Formatter::new();
#[inline]
pub fn bold(text: impl Display) -> String {
FORMATTER.bold(text)
}
#[inline]
pub fn italic(text: impl Display) -> String {
FORMATTER.italic(text)
}
#[inline]
pub fn underline(text: impl Display) -> String {
FORMATTER.underline(text)
}
#[inline]
pub fn strikethrough(text: impl Display) -> String {
FORMATTER.strikethrough(text)
}
#[inline]
pub fn spoiler(text: impl Display) -> String {
FORMATTER.spoiler(text)
}
#[inline]
pub fn blockquote(text: impl Display) -> String {
FORMATTER.blockquote(text)
}
#[inline]
pub fn expandable_blockquote(text: impl Display) -> String {
FORMATTER.expandable_blockquote(text)
}
#[inline]
pub fn text_link(text: impl Display, url: &str) -> String {
FORMATTER.text_link(text, url)
}
#[inline]
pub fn text_mention(text: impl Display, user_id: i64) -> String {
FORMATTER.text_mention(text, user_id)
}
#[inline]
pub fn custom_emoji(text: impl Display, emoji_id: &str) -> String {
FORMATTER.custom_emoji(text, emoji_id)
}
#[inline]
pub fn code(text: impl Display) -> String {
FORMATTER.code(text)
}
#[inline]
pub fn pre(text: impl Display) -> String {
FORMATTER.pre(text)
}
#[inline]
pub fn pre_language(text: impl Display, language: &str) -> String {
FORMATTER.pre_language(text, language)
}
#[inline]
pub fn date_time(text: impl Display, unix_time: i64) -> String {
FORMATTER.date_time(text, unix_time)
}
#[inline]
pub fn date_time_with_format(
text: impl Display,
unix_time: i64,
date_time_format: impl Display,
) -> String {
FORMATTER.date_time_with_format(text, unix_time, date_time_format)
}
#[inline]
pub fn quote(text: impl Display) -> String {
FORMATTER.quote(text)
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_bold() {
let formatter = Formatter;
assert_eq!(formatter.bold("text"), "*text*");
}
#[test]
fn test_italic() {
let formatter = Formatter;
assert_eq!(formatter.italic("text"), "_\rtext_\r");
}
#[test]
fn test_underline() {
let formatter = Formatter;
assert_eq!(formatter.underline("text"), "__\rtext__\r");
}
#[test]
fn test_strikethrough() {
let formatter = Formatter;
assert_eq!(formatter.strikethrough("text"), "~text~");
}
#[test]
fn test_spoiler() {
let formatter = Formatter;
assert_eq!(formatter.spoiler("text"), "|text|");
}
#[test]
fn test_blockquote() {
let formatter = Formatter;
assert_eq!(formatter.blockquote("text"), ">text");
assert_eq!(formatter.blockquote("text\ntext"), ">text\n>text");
}
#[test]
fn expandable_blockquote() {
let formatter = Formatter;
assert_eq!(formatter.expandable_blockquote("text"), ">text||");
assert_eq!(
formatter.expandable_blockquote("text\ntext"),
">text\n>text||"
);
}
#[test]
fn test_text_link() {
let formatter = Formatter;
assert_eq!(
formatter.text_link("text", "http://example.com"),
"[text](http://example.com)"
);
}
#[test]
fn test_text_mention() {
let formatter = Formatter;
assert_eq!(formatter.text_mention("text", 1), "[text](tg://user?id=1)");
}
#[test]
fn test_custom_emoji() {
let formatter = Formatter;
assert_eq!(
formatter.custom_emoji("text", "1"),
""
);
}
#[test]
fn test_code() {
let formatter = Formatter;
assert_eq!(formatter.code("text"), "`text`");
}
#[test]
fn test_pre() {
let formatter = Formatter;
assert_eq!(formatter.pre("text"), "```\ntext\n```");
}
#[test]
fn test_pre_language() {
let formatter = Formatter;
assert_eq!(
formatter.pre_language("text", "python"),
"```python\ntext\n```"
);
}
#[test]
fn test_date_time() {
let formatter = Formatter::default();
assert_eq!(formatter.date_time("text", 1), "");
assert_eq!(
formatter.date_time_with_format("text", 1, "test"),
""
);
}
#[test]
fn test_quote() {
let formatter = Formatter;
assert_eq!(formatter.quote("test"), "test");
assert_eq!(formatter.quote("[test]"), r"\[test\]");
assert_eq!(formatter.quote("test ` test"), r"test \` test");
assert_eq!(formatter.quote("test * test"), r"test \* test");
assert_eq!(formatter.quote("test _ test"), r"test \_ test");
assert_eq!(formatter.quote("test ~ test"), r"test \~ test");
assert_eq!(formatter.quote("test | test"), r"test \| test");
assert_eq!(formatter.quote("test > test"), r"test \> test");
assert_eq!(formatter.quote("test # test"), r"test \# test");
assert_eq!(formatter.quote("test + test"), r"test \+ test");
assert_eq!(formatter.quote("test - test"), r"test \- test");
assert_eq!(formatter.quote("test = test"), r"test \= test");
assert_eq!(formatter.quote("test . test"), r"test \. test");
assert_eq!(formatter.quote("test ! test"), r"test \! test");
assert_eq!(formatter.quote("test [ test"), r"test \[ test");
assert_eq!(formatter.quote("test ] test"), r"test \] test");
assert_eq!(formatter.quote("test ( test"), r"test \( test");
assert_eq!(formatter.quote("test ) test"), r"test \) test");
assert_eq!(formatter.quote("test { test"), r"test \{ test");
assert_eq!(formatter.quote("test } test"), r"test \} test");
assert_eq!(
formatter.quote("test ` * _ ~ | > # + - = . ! [ ] ( ) { } test"),
r"test \` \* \_ \~ \| \> \# \+ \- \= \. \! \[ \] \( \) \{ \} test"
);
}
}