use chrono::{NaiveDate, NaiveTime}; use std::collections::HashMap; use uuid::Uuid; use crate::model::repetition::Repetition; use crate::model::time; use crate::validation; pub static DATE_FORMAT: &str = "%d/%m/%Y"; #[derive(Debug, Clone)] pub struct Event { pub id: Uuid, pub date: NaiveDate, pub start: Option<NaiveTime>, pub end: Option<NaiveTime>, pub name: String, pub repetition: Option<Repetition>, } impl Event { pub fn pprint(&self) -> String { let start = self.start.map(time::pprint).unwrap_or_default(); let end = self .end .map(|t| format!("-{}", time::pprint(t))) .unwrap_or_default(); let space = if self.start.is_some() || self.end.is_some() { " " } else { "" }; format!("{}{}{}{}", start, end, space, self.name) } } /// Recurring events in an date range (inclusive) pub fn repetitions_between( events: &[Event], start: NaiveDate, end: NaiveDate, ) -> HashMap<NaiveDate, Vec<Event>> { let mut res: HashMap<NaiveDate, Vec<Event>> = HashMap::new(); for event in events { if let Some(repetition) = &event.repetition { for date in repetition.between(event.date, start, end) { res.entry(date).or_insert_with(Vec::new).push(event.clone()) } } } res } // Validation pub fn validate( id: Uuid, date: String, name: String, start: String, end: String, repetition: Option<Repetition>, ) -> Option<Event> { let start = validation::time(start)?; let end = validation::time(end)?; match (start, end) { (Some(s), Some(e)) if s > e => None?, _ => (), } Some(Event { id, date: NaiveDate::parse_from_str(&date, DATE_FORMAT).ok()?, name: validation::non_empty(name)?, start, end, repetition, }) }