use crate::blocks::elements::{
Checkboxes, DatePicker, DatetimePicker, EmailInput, FileInput, MultiSelectMenuConversations,
MultiSelectMenuExternalDataSource, MultiSelectMenuPublicChannels, MultiSelectMenuStaticOptions,
MultiSelectMenuUsers, NumberInput, PlainTextInput, RadioButtonGroup, RichTextInput,
SelectMenuConversations, SelectMenuExternalDataSource, SelectMenuPublicChannels,
SelectMenuStaticOptions, SelectMenuUsers, TimePicker, UrlInput,
};
use crate::composition_objects::{Plain, Text};
use crate::validators::*;
use serde::Serialize;
use slack_messaging_derive::Builder;
#[derive(Debug, Clone, Serialize, PartialEq, Builder)]
#[serde(tag = "type", rename = "input")]
pub struct Input {
#[builder(validate("required", "text_object::max_2000"))]
pub(crate) label: Option<Text<Plain>>,
#[builder(validate("required"))]
pub(crate) element: Option<InputElement>,
#[serde(skip_serializing_if = "Option::is_none")]
pub(crate) dispatch_action: Option<bool>,
#[serde(skip_serializing_if = "Option::is_none")]
#[builder(validate("text::max_255"))]
pub(crate) block_id: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
#[builder(validate("text_object::max_2000"))]
pub(crate) hint: Option<Text<Plain>>,
#[serde(skip_serializing_if = "Option::is_none")]
pub(crate) optional: Option<bool>,
}
#[derive(Debug, Clone, Serialize, PartialEq)]
#[serde(untagged)]
pub enum InputElement {
Checkboxes(Box<Checkboxes>),
DatePicker(Box<DatePicker>),
DatetimePicker(Box<DatetimePicker>),
EmailInput(Box<EmailInput>),
FileInput(Box<FileInput>),
MultiSelectMenuStaticOptions(Box<MultiSelectMenuStaticOptions>),
MultiSelectMenuExternalDataSource(Box<MultiSelectMenuExternalDataSource>),
MultiSelectMenuUsers(Box<MultiSelectMenuUsers>),
MultiSelectMenuConversations(Box<MultiSelectMenuConversations>),
MultiSelectMenuPublicChannels(Box<MultiSelectMenuPublicChannels>),
NumberInput(Box<NumberInput>),
PlainTextInput(Box<PlainTextInput>),
RadioButtonGroup(Box<RadioButtonGroup>),
RichTextInput(Box<RichTextInput>),
SelectMenuStaticOptions(Box<SelectMenuStaticOptions>),
SelectMenuExternalDataSource(Box<SelectMenuExternalDataSource>),
SelectMenuUsers(Box<SelectMenuUsers>),
SelectMenuConversations(Box<SelectMenuConversations>),
SelectMenuPublicChannels(Box<SelectMenuPublicChannels>),
TimePicker(Box<TimePicker>),
UrlInput(Box<UrlInput>),
}
macro_rules! input_from {
($($ty:ident,)*) => {
$(
impl From<$ty> for InputElement {
fn from(value: $ty) -> Self {
Self::$ty(Box::new(value))
}
}
)*
}
}
input_from! {
Checkboxes,
DatePicker,
DatetimePicker,
EmailInput,
FileInput,
MultiSelectMenuStaticOptions,
MultiSelectMenuExternalDataSource,
MultiSelectMenuUsers,
MultiSelectMenuConversations,
MultiSelectMenuPublicChannels,
NumberInput,
PlainTextInput,
RadioButtonGroup,
RichTextInput,
SelectMenuStaticOptions,
SelectMenuExternalDataSource,
SelectMenuUsers,
SelectMenuConversations,
SelectMenuPublicChannels,
TimePicker,
UrlInput,
}
#[cfg(test)]
mod tests {
use super::*;
use crate::blocks::elements::test_helpers::*;
use crate::composition_objects::test_helpers::*;
use crate::errors::*;
#[test]
fn it_implements_builder() {
let expected = Input {
label: Some(plain_text("foo")),
element: Some(text_input().into()),
dispatch_action: Some(true),
block_id: Some("input_0".into()),
hint: Some(plain_text("bar")),
optional: Some(true),
};
let val = Input::builder()
.set_label(Some(plain_text("foo")))
.set_element(Some(text_input()))
.set_dispatch_action(Some(true))
.set_block_id(Some("input_0"))
.set_hint(Some(plain_text("bar")))
.set_optional(Some(true))
.build()
.unwrap();
assert_eq!(val, expected);
let val = Input::builder()
.label(plain_text("foo"))
.element(text_input())
.dispatch_action(true)
.block_id("input_0")
.hint(plain_text("bar"))
.optional(true)
.build()
.unwrap();
assert_eq!(val, expected);
}
#[test]
fn it_requires_label_field() {
let err = Input::builder().element(text_input()).build().unwrap_err();
assert_eq!(err.object(), "Input");
let errors = err.field("label");
assert!(errors.includes(ValidationErrorKind::Required));
}
#[test]
fn it_requires_label_less_than_2000_characters_long() {
let err = Input::builder()
.label(plain_text("a".repeat(2001)))
.element(text_input())
.build()
.unwrap_err();
assert_eq!(err.object(), "Input");
let errors = err.field("label");
assert!(errors.includes(ValidationErrorKind::MaxTextLength(2000)));
}
#[test]
fn it_requires_element_field() {
let err = Input::builder()
.label(plain_text("foo"))
.build()
.unwrap_err();
assert_eq!(err.object(), "Input");
let errors = err.field("element");
assert!(errors.includes(ValidationErrorKind::Required));
}
#[test]
fn it_requires_block_id_less_than_255_characters_long() {
let err = Input::builder()
.label(plain_text("foo"))
.element(text_input())
.block_id("a".repeat(256))
.build()
.unwrap_err();
assert_eq!(err.object(), "Input");
let errors = err.field("block_id");
assert!(errors.includes(ValidationErrorKind::MaxTextLength(255)));
}
#[test]
fn it_requires_hint_less_then_2000_characters_long() {
let err = Input::builder()
.label(plain_text("foo"))
.element(text_input())
.hint(plain_text("a".repeat(2001)))
.build()
.unwrap_err();
assert_eq!(err.object(), "Input");
let errors = err.field("hint");
assert!(errors.includes(ValidationErrorKind::MaxTextLength(2000)));
}
}