slack_messaging/blocks/elements/
time_picker.rs1use crate::composition_objects::{ConfirmationDialog, Plain, Text};
2use crate::validators::*;
3
4use serde::Serialize;
5use slack_messaging_derive::Builder;
6
7#[derive(Debug, Clone, Serialize, PartialEq, Builder)]
68#[serde(tag = "type", rename = "timepicker")]
69pub struct TimePicker {
70 #[serde(skip_serializing_if = "Option::is_none")]
71 #[builder(validate("text::max_255"))]
72 pub(crate) action_id: Option<String>,
73
74 #[serde(skip_serializing_if = "Option::is_none")]
75 #[builder(validate("text::time_format"))]
76 pub(crate) initial_time: Option<String>,
77
78 #[serde(skip_serializing_if = "Option::is_none")]
79 pub(crate) confirm: Option<ConfirmationDialog>,
80
81 #[serde(skip_serializing_if = "Option::is_none")]
82 pub(crate) focus_on_load: Option<bool>,
83
84 #[serde(skip_serializing_if = "Option::is_none")]
85 #[builder(validate("text_object::max_150"))]
86 pub(crate) placeholder: Option<Text<Plain>>,
87
88 #[serde(skip_serializing_if = "Option::is_none")]
89 pub(crate) timezone: Option<String>,
90}
91
92#[cfg(test)]
93mod tests {
94 use super::*;
95 use crate::composition_objects::test_helpers::*;
96 use crate::errors::*;
97
98 #[test]
99 fn it_implements_builder() {
100 let expected = TimePicker {
101 action_id: Some("time_picker_0".into()),
102 initial_time: Some("10:30".into()),
103 confirm: Some(confirm()),
104 focus_on_load: Some(true),
105 placeholder: Some(plain_text("Select a time")),
106 timezone: Some("Asia/Tokyo".into()),
107 };
108
109 let val = TimePicker::builder()
110 .set_action_id(Some("time_picker_0"))
111 .set_initial_time(Some("10:30"))
112 .set_confirm(Some(confirm()))
113 .set_focus_on_load(Some(true))
114 .set_placeholder(Some(plain_text("Select a time")))
115 .set_timezone(Some("Asia/Tokyo"))
116 .build()
117 .unwrap();
118
119 assert_eq!(val, expected);
120
121 let val = TimePicker::builder()
122 .action_id("time_picker_0")
123 .initial_time("10:30")
124 .confirm(confirm())
125 .focus_on_load(true)
126 .placeholder(plain_text("Select a time"))
127 .timezone("Asia/Tokyo")
128 .build()
129 .unwrap();
130
131 assert_eq!(val, expected);
132 }
133
134 #[test]
135 fn it_requires_action_id_less_than_255_characters_long() {
136 let err = TimePicker::builder()
137 .action_id("a".repeat(256))
138 .build()
139 .unwrap_err();
140 assert_eq!(err.object(), "TimePicker");
141
142 let errors = err.field("action_id");
143 assert!(errors.includes(ValidationErrorKind::MaxTextLength(255)));
144 }
145
146 #[test]
147 fn it_requires_initial_time_matches_time_format() {
148 let err = TimePicker::builder()
149 .initial_time("foobar")
150 .build()
151 .unwrap_err();
152 assert_eq!(err.object(), "TimePicker");
153
154 let errors = err.field("initial_time");
155 assert!(errors.includes(ValidationErrorKind::InvalidFormat("24-hour format HH:mm")));
156 }
157
158 #[test]
159 fn it_requires_placeholder_less_than_150_characters_long() {
160 let err = TimePicker::builder()
161 .placeholder(plain_text("a".repeat(151)))
162 .build()
163 .unwrap_err();
164 assert_eq!(err.object(), "TimePicker");
165
166 let errors = err.field("placeholder");
167 assert!(errors.includes(ValidationErrorKind::MaxTextLength(150)));
168 }
169}