use chrono::Datelike;
use crate::eval::coercion::to_number;
use crate::eval::functions::check_arity;
use crate::eval::functions::date::serial::serial_to_date;
use crate::types::{ErrorKind, Value};
pub fn weeknum_fn(args: &[Value]) -> Value {
if let Some(err) = check_arity(args, 1, 2) {
return err;
}
let serial = match to_number(args[0].clone()) { Ok(n) => n, Err(e) => return e };
let return_type = if args.len() > 1 {
match to_number(args[1].clone()) { Ok(n) => n, Err(e) => return e }
} else {
1.0
};
let date = match serial_to_date(serial) {
Some(d) => d,
None => return Value::Error(ErrorKind::Value),
};
if return_type as u32 == 21 {
return Value::Number(date.iso_week().week() as f64);
}
let jan1 = chrono::NaiveDate::from_ymd_opt(date.year(), 1, 1).unwrap();
let day_of_year = date.ordinal() as i32 - 1;
let offset = match return_type as u32 {
1 | 17 => jan1.weekday().num_days_from_sunday() as i32,
2 | 11 => jan1.weekday().num_days_from_monday() as i32,
12 => (jan1.weekday().num_days_from_monday() as i32 + 6) % 7, 13 => (jan1.weekday().num_days_from_monday() as i32 + 5) % 7, 14 => (jan1.weekday().num_days_from_monday() as i32 + 4) % 7, 15 => (jan1.weekday().num_days_from_monday() as i32 + 3) % 7, 16 => (jan1.weekday().num_days_from_monday() as i32 + 2) % 7, _ => return Value::Error(ErrorKind::Num),
};
let week = (day_of_year + offset) / 7 + 1;
Value::Number(week as f64)
}
#[cfg(test)]
mod tests;