1use crate::aes::Aesthetic;
2use crate::data::Value;
3
4use super::Scale;
5
6#[derive(Clone, Copy, Debug)]
8enum DateBreak {
9 Secs(f64),
11 Months(u32),
13}
14
15#[derive(Clone, Debug)]
17pub struct ScaleDateTime {
18 aesthetic: Aesthetic,
19 name: String,
20 min: f64,
21 max: f64,
22 trained: bool,
23 expand: (f64, f64),
24 date_breaks: Option<DateBreak>,
25 date_labels: Option<String>,
26}
27
28struct DateParts {
30 year: i64,
31 month: u32,
32 day: u32,
33 hour: u32,
34 minute: u32,
35 second: u32,
36}
37
38fn days_from_civil(y: i64, m: u32, d: u32) -> i64 {
40 let y = if m <= 2 { y - 1 } else { y };
41 let era = (if y >= 0 { y } else { y - 399 }) / 400;
42 let yoe = y - era * 400;
43 let mp = if m > 2 { m - 3 } else { m + 9 } as i64;
44 let doy = (153 * mp + 2) / 5 + d as i64 - 1;
45 let doe = yoe * 365 + yoe / 4 - yoe / 100 + doy;
46 era * 146_097 + doe - 719_468
47}
48
49fn secs_from_civil(y: i64, m: u32, d: u32) -> i64 {
50 days_from_civil(y, m, d) * 86_400
51}
52
53fn civil_from_secs(secs: i64) -> DateParts {
55 let (mut days, rem) = if secs >= 0 {
56 (secs / 86_400, secs % 86_400)
57 } else {
58 let d = (secs - 86_400 + 1) / 86_400;
59 (d, secs - d * 86_400)
60 };
61 let hour = (rem / 3600) as u32;
62 let minute = ((rem % 3600) / 60) as u32;
63 let second = (rem % 60) as u32;
64
65 days += 719_468;
66 let era = if days >= 0 { days } else { days - 146_096 } / 146_097;
67 let doe = (days - era * 146_097) as u32;
68 let yoe = (doe - doe / 1460 + doe / 36524 - doe / 146_096) / 365;
69 let y = yoe as i64 + era * 400;
70 let doy = doe - (365 * yoe + yoe / 4 - yoe / 100);
71 let mp = (5 * doy + 2) / 153;
72 let day = doy - (153 * mp + 2) / 5 + 1;
73 let month = if mp < 10 { mp + 3 } else { mp - 9 };
74 let year = if month <= 2 { y + 1 } else { y };
75 DateParts {
76 year,
77 month,
78 day,
79 hour,
80 minute,
81 second,
82 }
83}
84
85const MONTHS_SHORT: [&str; 12] = [
86 "Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec",
87];
88const MONTHS_LONG: [&str; 12] = [
89 "January",
90 "February",
91 "March",
92 "April",
93 "May",
94 "June",
95 "July",
96 "August",
97 "September",
98 "October",
99 "November",
100 "December",
101];
102
103fn strftime(secs: f64, fmt: &str) -> String {
106 let p = civil_from_secs(secs as i64);
107 let mi = (p.month.clamp(1, 12) - 1) as usize;
108 let mut out = String::new();
109 let mut chars = fmt.chars();
110 while let Some(c) = chars.next() {
111 if c != '%' {
112 out.push(c);
113 continue;
114 }
115 match chars.next() {
116 Some('Y') => out.push_str(&format!("{:04}", p.year)),
117 Some('y') => out.push_str(&format!("{:02}", p.year.rem_euclid(100))),
118 Some('m') => out.push_str(&format!("{:02}", p.month)),
119 Some('b') => out.push_str(MONTHS_SHORT[mi]),
120 Some('B') => out.push_str(MONTHS_LONG[mi]),
121 Some('d') => out.push_str(&format!("{:02}", p.day)),
122 Some('e') => out.push_str(&format!("{:2}", p.day)),
123 Some('H') => out.push_str(&format!("{:02}", p.hour)),
124 Some('M') => out.push_str(&format!("{:02}", p.minute)),
125 Some('S') => out.push_str(&format!("{:02}", p.second)),
126 Some('%') => out.push('%'),
127 Some(other) => {
128 out.push('%');
129 out.push(other);
130 }
131 None => out.push('%'),
132 }
133 }
134 out
135}
136
137fn parse_date_break(spec: &str) -> Option<DateBreak> {
139 let spec = spec.trim().to_lowercase();
140 let mut parts = spec.split_whitespace();
141 let first = parts.next()?;
142 let (n, unit) = match first.parse::<f64>() {
143 Ok(n) => (n, parts.next()?.to_string()),
144 Err(_) => (1.0, first.to_string()),
145 };
146 let unit = unit.trim_end_matches('s');
147 let secs = |s: f64| Some(DateBreak::Secs(n * s));
148 match unit {
149 "sec" | "second" => secs(1.0),
150 "min" | "minute" => secs(60.0),
151 "hour" => secs(3600.0),
152 "day" => secs(86_400.0),
153 "week" => secs(604_800.0),
154 "month" => Some(DateBreak::Months(n.max(1.0) as u32)),
155 "year" => Some(DateBreak::Months((n.max(1.0) as u32) * 12)),
156 _ => None,
157 }
158}
159
160impl ScaleDateTime {
161 pub fn new() -> Self {
162 ScaleDateTime {
163 aesthetic: Aesthetic::X,
164 name: String::new(),
165 min: f64::INFINITY,
166 max: f64::NEG_INFINITY,
167 trained: false,
168 expand: (0.05, 0.0),
169 date_breaks: None,
170 date_labels: None,
171 }
172 }
173
174 pub fn for_aesthetic(mut self, aes: Aesthetic) -> Self {
175 self.aesthetic = aes;
176 self
177 }
178
179 pub fn with_name(mut self, name: &str) -> Self {
180 self.name = name.to_string();
181 self
182 }
183
184 pub fn with_date_breaks(mut self, spec: &str) -> Self {
187 self.date_breaks = parse_date_break(spec);
188 self
189 }
190
191 pub fn with_date_labels(mut self, fmt: &str) -> Self {
194 self.date_labels = Some(fmt.to_string());
195 self
196 }
197
198 fn label(&self, secs: f64, step: f64) -> String {
199 match &self.date_labels {
200 Some(fmt) => strftime(secs, fmt),
201 None => Self::format_datetime(secs, step),
202 }
203 }
204
205 fn expanded_range(&self) -> (f64, f64) {
206 let range = self.max - self.min;
207 let mult = self.expand.0;
208 let add = self.expand.1;
209 (self.min - range * mult - add, self.max + range * mult + add)
210 }
211
212 fn nice_datetime_step(range_secs: f64) -> f64 {
214 const MINUTE: f64 = 60.0;
215 const HOUR: f64 = 3600.0;
216 const DAY: f64 = 86400.0;
217 const WEEK: f64 = 7.0 * DAY;
218 const MONTH: f64 = 30.0 * DAY;
219 const YEAR: f64 = 365.25 * DAY;
220
221 let candidates = [
222 1.0,
223 5.0,
224 10.0,
225 30.0,
226 MINUTE,
227 5.0 * MINUTE,
228 10.0 * MINUTE,
229 30.0 * MINUTE,
230 HOUR,
231 3.0 * HOUR,
232 6.0 * HOUR,
233 12.0 * HOUR,
234 DAY,
235 2.0 * DAY,
236 WEEK,
237 2.0 * WEEK,
238 MONTH,
239 3.0 * MONTH,
240 6.0 * MONTH,
241 YEAR,
242 2.0 * YEAR,
243 5.0 * YEAR,
244 10.0 * YEAR,
245 20.0 * YEAR,
246 50.0 * YEAR,
247 100.0 * YEAR,
248 ];
249
250 let target = range_secs / 5.0;
251 for &c in &candidates {
252 if c >= target {
253 return c;
254 }
255 }
256 let n = (target / (100.0 * YEAR)).ceil();
258 n * 100.0 * YEAR
259 }
260
261 fn format_datetime(secs: f64, _step: f64) -> String {
264 let epoch_secs = secs as i64;
265 crate::data::format_epoch_secs(epoch_secs)
266 }
267}
268
269impl Default for ScaleDateTime {
270 fn default() -> Self {
271 Self::new()
272 }
273}
274
275impl Scale for ScaleDateTime {
276 fn aesthetic(&self) -> Aesthetic {
277 self.aesthetic.clone()
278 }
279
280 fn train(&mut self, values: &[Value]) {
281 for v in values {
282 if let Some(f) = v.as_f64() {
283 if f.is_finite() {
284 if f < self.min {
285 self.min = f;
286 }
287 if f > self.max {
288 self.max = f;
289 }
290 }
291 }
292 }
293 self.trained = true;
294 }
295
296 fn map(&self, value: &Value) -> f64 {
297 let f = match value.as_f64() {
298 Some(f) => f,
299 None => return 0.0,
300 };
301 let (emin, emax) = self.expanded_range();
302 let range = emax - emin;
303 if range.abs() < f64::EPSILON {
304 0.5
305 } else {
306 (f - emin) / range
307 }
308 }
309
310 fn breaks(&self) -> Vec<(f64, String)> {
311 if !self.trained || self.min > self.max {
312 return vec![];
313 }
314
315 let range = self.max - self.min;
316 if range.abs() < f64::EPSILON {
317 return vec![(0.5, self.label(self.min, 1.0))];
318 }
319
320 let (emin, emax) = self.expanded_range();
321
322 if let Some(DateBreak::Months(n)) = self.date_breaks {
324 let n = n.max(1);
325 let start = civil_from_secs(emin.ceil() as i64);
326 let (mut y, mut m) = (start.year, start.month);
327 if (secs_from_civil(y, m, 1) as f64) < emin {
329 m += 1;
330 if m > 12 {
331 m = 1;
332 y += 1;
333 }
334 }
335 let mut breaks = Vec::new();
336 let mut guard = 0;
337 loop {
338 let secs = secs_from_civil(y, m, 1) as f64;
339 if secs > emax + 1.0 || guard > 10_000 {
340 break;
341 }
342 breaks.push((self.map(&Value::Float(secs)), self.label(secs, 0.0)));
343 m += n;
344 while m > 12 {
345 m -= 12;
346 y += 1;
347 }
348 guard += 1;
349 }
350 return breaks;
351 }
352
353 let step = match self.date_breaks {
354 Some(DateBreak::Secs(s)) if s > 0.0 => s,
355 _ => Self::nice_datetime_step(range),
356 };
357 let start = (emin / step).ceil() * step;
358 let mut breaks = Vec::new();
359 let mut v = start;
360 while v <= emax + step * 0.001 {
361 breaks.push((self.map(&Value::Float(v)), self.label(v, step)));
362 v += step;
363 }
364 breaks
365 }
366
367 fn name(&self) -> &str {
368 &self.name
369 }
370
371 fn set_name(&mut self, name: &str) {
372 self.name = name.to_string();
373 }
374
375 fn set_limits(&mut self, min: f64, max: f64) {
376 self.min = min;
377 self.max = max;
378 self.trained = true;
379 }
380
381 fn clone_box(&self) -> Box<dyn Scale> {
382 Box::new(self.clone())
383 }
384
385 fn reset_training(&mut self) {
386 self.min = f64::INFINITY;
387 self.max = f64::NEG_INFINITY;
388 self.trained = false;
389 }
390}
391
392#[cfg(test)]
393mod tests {
394 use super::*;
395
396 #[test]
397 fn civil_roundtrip() {
398 let p = civil_from_secs(1_615_811_445);
400 assert_eq!((p.year, p.month, p.day), (2021, 3, 15));
401 assert_eq!((p.hour, p.minute, p.second), (12, 30, 45));
402 assert_eq!(secs_from_civil(2021, 3, 15), 1_615_766_400); }
404
405 #[test]
406 fn strftime_subset() {
407 let s = 1_615_766_400.0; assert_eq!(strftime(s, "%Y-%m-%d"), "2021-03-15");
409 assert_eq!(strftime(s, "%b %Y"), "Mar 2021");
410 assert_eq!(strftime(s, "%B"), "March");
411 assert_eq!(strftime(s, "100%%"), "100%");
412 }
413
414 #[test]
415 fn parse_specs() {
416 assert!(matches!(
417 parse_date_break("1 month"),
418 Some(DateBreak::Months(1))
419 ));
420 assert!(matches!(
421 parse_date_break("3 months"),
422 Some(DateBreak::Months(3))
423 ));
424 assert!(matches!(
425 parse_date_break("1 year"),
426 Some(DateBreak::Months(12))
427 ));
428 assert!(
429 matches!(parse_date_break("2 weeks"), Some(DateBreak::Secs(s)) if s == 1_209_600.0)
430 );
431 assert!(matches!(
432 parse_date_break("month"),
433 Some(DateBreak::Months(1))
434 ));
435 assert!(parse_date_break("fortnight").is_none());
436 }
437
438 #[test]
439 fn monthly_breaks_land_on_first_of_month() {
440 let mut s = ScaleDateTime::new()
441 .with_date_breaks("1 month")
442 .with_date_labels("%Y-%m-%d");
443 s.set_limits(
445 secs_from_civil(2021, 1, 10) as f64,
446 secs_from_civil(2021, 4, 20) as f64,
447 );
448 let labels: Vec<String> = s.breaks().into_iter().map(|(_, l)| l).collect();
449 assert!(labels.iter().all(|l| l.ends_with("-01")), "{labels:?}");
450 assert!(labels.contains(&"2021-02-01".to_string()));
451 }
452}