Skip to content

Commit

Permalink
Add datetime value parser
Browse files Browse the repository at this point in the history
Resolves greshake#1708
  • Loading branch information
bim9262 committed Feb 11, 2023
1 parent efa7b55 commit 35b8f4f
Show file tree
Hide file tree
Showing 4 changed files with 128 additions and 37 deletions.
57 changes: 23 additions & 34 deletions src/blocks/time.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,10 +4,9 @@
//!
//! Key | Values | Default
//! ----|--------|--------
//! `format` | Format string. See [chrono docs](https://docs.rs/chrono/0.3.0/chrono/format/strftime/index.html#specifiers) for all options. | `" $icon %a %d/%m %R "`
//! `format` | Format string. See [chrono docs](https://docs.rs/chrono/0.3.0/chrono/format/strftime/index.html#specifiers) for all options. | `" $icon $timestamp.datetime() "`
//! `interval` | Update interval in seconds | `10`
//! `timezone` | A timezone specifier (e.g. "Europe/Lisbon") | Local timezone
//! `locale` | Locale to apply when formatting the time | System locale
//!
//! Placeholder | Value | Type | Unit
//! --------------|---------------------------------------------|--------|-----
Expand All @@ -21,15 +20,13 @@
//! interval = 60
//! locale = "fr_BE"
//! [block.format]
//! full = " $icon %d/%m %R "
//! short = " $icon %R "
//! full = " $icon $timestamp.datetime(f:%a %Y-%m-%d %R %Z, l:fr_BE) "
//! short = " $icon $timestamp.datetime(f:%R) "
//! ```
//!
//! # Icons Used
//! - `time`
use chrono::offset::{Local, Utc};
use chrono::Locale;
use chrono_tz::Tz;

use super::prelude::*;
Expand All @@ -52,14 +49,27 @@ pub async fn run(config: Config, mut api: CommonApi) -> Result<()> {
.format
.full
.as_deref()
.unwrap_or(" $icon %a %d/%m %R ");
let format_short = config.format.short.as_deref();
.unwrap_or(" $icon $timestamp.datetime() ");

let format_short = config.format.short.as_deref().unwrap_or_default();

widget.set_format(FormatConfig::default().with_defaults(format, format_short)?);

let timezone = config.timezone;
let locale = match config.locale.as_deref() {
Some(locale) => Some(locale.try_into().ok().error("invalid locale")?),
None => None,
};

// let (full_time_format, short_time_format) = match config.locale.as_deref() {
// Some(locale) => {
// let locale = locale.try_into().ok().error("invalid locale")?;
// (
// StrftimeItems::new_with_locale(format, locale),
// format_short.map(|format| StrftimeItems::new_with_locale(format, locale)),
// )
// }
// None => (
// StrftimeItems::new(format),
// format_short.map(StrftimeItems::new),
// ),
// };

let mut timer = config.interval.timer();

Expand All @@ -70,13 +80,8 @@ pub async fn run(config: Config, mut api: CommonApi) -> Result<()> {
unsafe { tzset() };
}

let full_time = get_time(format, timezone, locale);
let short_time = format_short
.map(|f| get_time(f, timezone, locale))
.unwrap_or_else(|| "".into());

widget.set_format(FormatConfig::default().with_defaults(&full_time, &short_time)?);
widget.set_values(map!("icon" => Value::icon(api.get_icon("time")?)));
widget.set_values(map!("timestamp" => Value::timestamp(timezone)));

api.set_widget(&widget).await?;

Expand All @@ -87,22 +92,6 @@ pub async fn run(config: Config, mut api: CommonApi) -> Result<()> {
}
}

fn get_time(format: &str, timezone: Option<Tz>, locale: Option<Locale>) -> String {
match locale {
Some(locale) => match timezone {
Some(tz) => Utc::now()
.with_timezone(&tz)
.format_localized(format, locale)
.to_string(),
None => Local::now().format_localized(format, locale).to_string(),
},
None => match timezone {
Some(tz) => Utc::now().with_timezone(&tz).format(format).to_string(),
None => Local::now().format(format).to_string(),
},
}
}

extern "C" {
/// The tzset function initializes the tzname variable from the value of the TZ environment
/// variable. It is not usually necessary for your program to call this function, because it is
Expand Down
98 changes: 96 additions & 2 deletions src/formatting/formatter.rs
Original file line number Diff line number Diff line change
@@ -1,3 +1,5 @@
use chrono::{Local, Locale, Utc};

use std::fmt::Debug;
use std::iter::repeat;
use std::time::{Duration, Instant};
Expand All @@ -18,6 +20,8 @@ const DEFAULT_BAR_MAX_VAL: f64 = 100.0;

const DEFAULT_NUMBER_WIDTH: usize = 2;

const DEFAULT_DATETIME_FORMAT: &str = "%a %d/%m %R";

pub const DEFAULT_STRING_FORMATTER: StrFormatter = StrFormatter {
min_width: DEFAULT_STR_MIN_WIDTH,
max_width: DEFAULT_STR_MAX_WIDTH,
Expand All @@ -37,6 +41,10 @@ pub const DEFAULT_NUMBER_FORMATTER: EngFormatter = EngFormatter(EngFixConfig {
prefix_forced: false,
});

pub const DEFAULT_TIMESTAMP_FORMATTER: TimestampFormatter = TimestampFormatter {
format: None,
locale: None,
};
pub const DEFAULT_FLAG_FORMATTER: FlagFormatter = FlagFormatter;

pub trait Formatter: Debug + Send + Sync {
Expand Down Expand Up @@ -112,14 +120,35 @@ pub fn new_formatter(name: &str, args: &[Arg]) -> Result<Box<dyn Formatter>> {
max_value = arg.val.parse().error("Max value must be a number")?;
}
other => {
return Err(Error::new(format!("Unknown argumnt for 'bar': '{other}'")));
return Err(Error::new(format!("Unknown argument for 'bar': '{other}'")));
}
}
}
Ok(Box::new(BarFormatter { width, max_value }))
}
"eng" => Ok(Box::new(EngFormatter(EngFixConfig::from_args(args)?))),
"fix" => Ok(Box::new(FixFormatter(EngFixConfig::from_args(args)?))),
"datetime" => {
let mut format = None;
let mut locale = None;
for arg in args {
match arg.key {
"format" | "f" => {
format = Some(arg.val.parse().error("format value must be a string")?);
}
"locale" | "l" => {
locale = arg.val.parse().ok();
}
other => {
return Err(Error::new(format!(
"Unknown argument for 'datetime': '{other}'"
)));
}
}
}

Ok(Box::new(TimestampFormatter::new(format, locale)?))
}
_ => Err(Error::new(format!("Unknown formatter: '{name}'"))),
}
}
Expand Down Expand Up @@ -163,6 +192,9 @@ impl Formatter for StrFormatter {
Value::Number { .. } => Err(Error::new_format(
"A number cannot be formatted with 'str' formatter",
)),
Value::Timestamp(..) => Err(Error::new_format(
"A timestamp cannot be formatted with 'str' formatter",
)),
Value::Flag => Err(Error::new_format(
"A flag cannot be formatted with 'str' formatter",
)),
Expand All @@ -184,6 +216,9 @@ impl Formatter for PangoStrFormatter {
Value::Number { .. } => Err(Error::new_format(
"A number cannot be formatted with 'str' formatter",
)),
Value::Timestamp(..) => Err(Error::new_format(
"A timestamp cannot be formatted with 'str' formatter",
)),
Value::Flag => Err(Error::new_format(
"A flag cannot be formatted with 'str' formatter",
)),
Expand Down Expand Up @@ -220,6 +255,9 @@ impl Formatter for BarFormatter {
Value::Icon(_) => Err(Error::new_format(
"An icon cannot be formatted with 'bar' formatter",
)),
Value::Timestamp(..) => Err(Error::new_format(
"A timestamp cannot be formatted with 'bar' formatter",
)),
Value::Flag => Err(Error::new_format(
"A flag cannot be formatted with 'bar' formatter",
)),
Expand Down Expand Up @@ -367,6 +405,9 @@ impl Formatter for EngFormatter {
Value::Icon(_) => Err(Error::new_format(
"An icon cannot be formatted with 'eng' formatter",
)),
Value::Timestamp(..) => Err(Error::new_format(
"A timestamp cannot be formatted with 'eng' formatter",
)),
Value::Flag => Err(Error::new_format(
"A flag cannot be formatted with 'eng' formatter",
)),
Expand All @@ -392,20 +433,73 @@ impl Formatter for FixFormatter {
Value::Icon(_) => Err(Error::new_format(
"An icon cannot be formatted with 'fix' formatter",
)),
Value::Timestamp(..) => Err(Error::new_format(
"A timestamp cannot be formatted with 'fix' formatter",
)),
Value::Flag => Err(Error::new_format(
"A flag cannot be formatted with 'fix' formatter",
)),
}
}
}

#[derive(Debug)]
pub struct TimestampFormatter {
format: Option<String>,
locale: Option<Locale>,
}

impl TimestampFormatter {
fn new(format: Option<String>, locale: Option<String>) -> Result<Self> {
let locale = match locale.as_deref() {
Some(locale) => Some(locale.try_into().ok().error("invalid locale")?),
None => None,
};
Ok(Self { format, locale })
}

fn get_format(&self) -> String {
self.format
.clone()
.unwrap_or(DEFAULT_DATETIME_FORMAT.to_string())
}
}

impl Formatter for TimestampFormatter {
fn format(&self, val: &Value) -> Result<String> {
match val {
Value::Number { .. } | Value::Text(_) | Value::Icon(_) | Value::Flag => unreachable!(),
Value::Timestamp(timezone) => Ok(match self.locale {
Some(locale) => match timezone {
Some(tz) => Utc::now()
.with_timezone(tz)
.format_localized(&self.get_format(), locale)
.to_string(),
None => Local::now()
.format_localized(&self.get_format(), locale)
.to_string(),
},
None => match timezone {
Some(tz) => Utc::now()
.with_timezone(tz)
.format(&self.get_format())
.to_string(),
None => Local::now().format(&self.get_format()).to_string(),
},
}),
}
}
}

#[derive(Debug)]
pub struct FlagFormatter;

impl Formatter for FlagFormatter {
fn format(&self, val: &Value) -> Result<String> {
match val {
Value::Number { .. } | Value::Text(_) | Value::Icon(_) => unreachable!(),
Value::Number { .. } | Value::Text(_) | Value::Icon(_) | Value::Timestamp(..) => {
unreachable!()
}
Value::Flag => Ok(String::new()),
}
}
Expand Down
4 changes: 3 additions & 1 deletion src/formatting/parse.rs
Original file line number Diff line number Diff line change
Expand Up @@ -82,7 +82,9 @@ fn alphanum1(i: &str) -> IResult<&str, &str, PError> {
}

fn arg1(i: &str) -> IResult<&str, &str, PError> {
take_while1(|x: char| x.is_alphanumeric() || x == '_' || x == '-' || x == '.')(i)
take_while1(|x: char| {
x.is_alphanumeric() || x == ' ' || x == '%' || x == '/' || x == '_' || x == '-' || x == '.'
})(i)
}

// `key:val`
Expand Down
6 changes: 6 additions & 0 deletions src/formatting/value.rs
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
use super::formatter;
use super::unit::Unit;
use super::Metadata;
use chrono_tz::Tz;

#[derive(Debug, Clone)]
pub struct Value {
Expand All @@ -13,6 +14,7 @@ pub enum ValueInner {
Text(String),
Icon(String),
Number { val: f64, unit: Unit },
Timestamp(Option<Tz>),
Flag,
}

Expand Down Expand Up @@ -46,6 +48,9 @@ impl Value {
Self::new(ValueInner::Flag)
}

pub fn timestamp(tz: Option<Tz>) -> Self {
Self::new(ValueInner::Timestamp(tz))
}
pub fn icon(icon: String) -> Self {
Self::new(ValueInner::Icon(icon))
}
Expand Down Expand Up @@ -108,6 +113,7 @@ impl Value {
match &self.inner {
ValueInner::Text(_) | ValueInner::Icon(_) => &formatter::DEFAULT_STRING_FORMATTER,
ValueInner::Number { .. } => &formatter::DEFAULT_NUMBER_FORMATTER,
ValueInner::Timestamp { .. } => &formatter::DEFAULT_TIMESTAMP_FORMATTER,
ValueInner::Flag => &formatter::DEFAULT_FLAG_FORMATTER,
}
}
Expand Down

0 comments on commit 35b8f4f

Please sign in to comment.