use crate::validators::*;
use paste::paste;
use serde::Serialize;
use slack_messaging_derive::Builder;
pub mod builders;
pub mod types;
mod list;
mod preformatted;
mod quote;
mod section;
pub use list::{ListStyle, RichTextList};
pub use preformatted::RichTextPreformatted;
pub use quote::RichTextQuote;
pub use section::RichTextSection;
#[derive(Debug, Clone, Serialize, PartialEq)]
#[serde(untagged)]
pub enum RichTextSubElement {
Section(Box<RichTextSection>),
List(Box<RichTextList>),
Preformatted(Box<RichTextPreformatted>),
Quote(Box<RichTextQuote>),
}
macro_rules! impl_sub_element {
($($var:tt,)*) => {
paste! {
$(
impl From<[<RichText $var>]> for RichTextSubElement {
fn from(value: [<RichText $var>]) -> Self {
Self::$var(Box::new(value))
}
}
)*
}
};
}
impl_sub_element! {
Section,
List,
Preformatted,
Quote,
}
#[derive(Debug, Clone, Serialize, PartialEq, Builder)]
#[serde(tag = "type", rename = "rich_text")]
pub struct RichText {
#[builder(push_item = "element", validate("required"))]
pub(crate) elements: Option<Vec<RichTextSubElement>>,
#[serde(skip_serializing_if = "Option::is_none")]
#[builder(validate("text::max_255"))]
pub(crate) block_id: Option<String>,
}
#[cfg(test)]
mod tests {
use super::test_helpers::*;
use super::types::test_helpers::*;
use super::*;
use crate::errors::*;
#[test]
fn it_implements_builder() {
let expected = RichText {
block_id: Some("rich_text_0".into()),
elements: Some(vec![section(vec![el_text("foo"), el_emoji("var")]).into()]),
};
let val = RichText::builder()
.set_block_id(Some("rich_text_0"))
.set_elements(Some(vec![
section(vec![el_text("foo"), el_emoji("var")]).into(),
]))
.build()
.unwrap();
assert_eq!(val, expected);
let val = RichText::builder()
.block_id("rich_text_0")
.elements(vec![section(vec![el_text("foo"), el_emoji("var")]).into()])
.build()
.unwrap();
assert_eq!(val, expected);
}
#[test]
fn it_implements_push_item_method() {
let expected = RichText {
block_id: None,
elements: Some(vec![
section(vec![el_text("foo"), el_emoji("var")]).into(),
section(vec![el_text("baz")]).into(),
]),
};
let val = RichText::builder()
.element(section(vec![el_text("foo"), el_emoji("var")]))
.element(section(vec![el_text("baz")]))
.build()
.unwrap();
assert_eq!(val, expected);
}
#[test]
fn it_requires_elements_field() {
let err = RichText::builder().build().unwrap_err();
assert_eq!(err.object(), "RichText");
let errors = err.field("elements");
assert!(errors.includes(ValidationErrorKind::Required));
}
#[test]
fn it_requires_block_id_less_than_255_characters_long() {
let err = RichText::builder()
.block_id("a".repeat(256))
.element(section(vec![el_text("baz")]))
.build()
.unwrap_err();
assert_eq!(err.object(), "RichText");
let errors = err.field("block_id");
assert!(errors.includes(ValidationErrorKind::MaxTextLength(255)));
}
}
#[cfg(test)]
pub mod test_helpers {
use super::types::RichTextElementType;
use super::*;
pub fn section(elements: Vec<RichTextElementType>) -> RichTextSection {
RichTextSection {
elements: Some(elements),
}
}
}