use chrono::DateTime;
use serde::{Deserialize, Serialize};
use std::fmt;
#[derive(Debug, Clone, Copy, PartialEq, Default)]
pub enum UserTimezone {
#[default]
Utc,
Local,
}
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum TimeLabelKind<'a> {
Axis { timeframe: exchange::Timeframe },
Crosshair { show_millis: bool },
Custom(&'a str),
}
impl UserTimezone {
pub fn to_user_datetime(
&self,
datetime: DateTime<chrono::Utc>,
) -> DateTime<chrono::FixedOffset> {
self.with_user_timezone(datetime, |time_with_zone| time_with_zone)
}
pub fn format_with_kind(&self, timestamp_ms: i64, kind: TimeLabelKind<'_>) -> Option<String> {
DateTime::from_timestamp_millis(timestamp_ms).map(|datetime| {
self.with_user_timezone(datetime, |time_with_zone| match kind {
TimeLabelKind::Axis { timeframe } => {
Self::format_by_timeframe(&time_with_zone, timeframe)
}
TimeLabelKind::Crosshair { show_millis } => {
if show_millis {
time_with_zone.format("%H:%M:%S.%3f").to_string()
} else {
time_with_zone.format("%a %b %-d %H:%M").to_string()
}
}
TimeLabelKind::Custom(fmt) => time_with_zone.format(fmt).to_string(),
})
})
}
fn with_user_timezone<T>(
&self,
datetime: DateTime<chrono::Utc>,
formatter: impl FnOnce(DateTime<chrono::FixedOffset>) -> T,
) -> T {
let time_with_zone = match self {
UserTimezone::Local => datetime.with_timezone(&chrono::Local).fixed_offset(),
UserTimezone::Utc => datetime.fixed_offset(),
};
formatter(time_with_zone)
}
fn format_by_timeframe(
datetime: &DateTime<chrono::FixedOffset>,
timeframe: exchange::Timeframe,
) -> String {
let interval = timeframe.to_milliseconds();
if interval < 10_000 {
datetime.format("%M:%S").to_string()
} else if datetime.format("%H:%M").to_string() == "00:00" {
datetime.format("%-d").to_string()
} else {
datetime.format("%H:%M").to_string()
}
}
}
impl fmt::Display for UserTimezone {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
UserTimezone::Utc => write!(f, "UTC"),
UserTimezone::Local => {
let local_offset = chrono::Local::now().offset().local_minus_utc();
let hours = local_offset / 3600;
let minutes = (local_offset % 3600) / 60;
write!(f, "Local (UTC {hours:+03}:{minutes:02})")
}
}
}
}
impl<'de> Deserialize<'de> for UserTimezone {
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
where
D: serde::Deserializer<'de>,
{
let timezone_str = String::deserialize(deserializer)?;
match timezone_str.to_lowercase().as_str() {
"utc" => Ok(UserTimezone::Utc),
"local" => Ok(UserTimezone::Local),
_ => Err(serde::de::Error::custom("Invalid UserTimezone")),
}
}
}
impl Serialize for UserTimezone {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: serde::Serializer,
{
match self {
UserTimezone::Utc => serializer.serialize_str("UTC"),
UserTimezone::Local => serializer.serialize_str("Local"),
}
}
}