From 49057cfdc6ff69114dbc5190d3ad70e26d74143c Mon Sep 17 00:00:00 2001 From: Akshay Date: Wed, 11 Mar 2020 18:22:07 +0530 Subject: flattened views module --- src/views.rs | 125 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 125 insertions(+) create mode 100644 src/views.rs (limited to 'src/views.rs') diff --git a/src/views.rs b/src/views.rs new file mode 100644 index 0000000..ebb83b8 --- /dev/null +++ b/src/views.rs @@ -0,0 +1,125 @@ +use cursive::direction::Direction; +use cursive::event::{Event, EventResult, Key}; +use cursive::theme::Style; +use cursive::view::View; +use cursive::{Printer, Vec2}; + +use chrono::prelude::*; +use chrono::{Local, NaiveDate}; + +use crate::habit::{Bit, Count, Habit, TrackEvent}; +use crate::CONFIGURATION; + +pub trait ShadowView { + fn draw(&self, printer: &Printer); + fn required_size(&mut self, _: Vec2) -> Vec2; + fn take_focus(&mut self, _: Direction) -> bool; + fn on_event(&mut self, e: Event) -> EventResult; +} + +// the only way to not rewrite each View implementation for trait +// objects of Habit is to rewrite the View trait itself. +impl ShadowView for T +where + T: Habit, + T::HabitType: std::fmt::Display, +{ + fn draw(&self, printer: &Printer) { + let now = Local::now(); + let year = now.year(); + let month = now.month(); + + let goal_reached_style = Style::from(CONFIGURATION.reached_color); + let todo_style = Style::from(CONFIGURATION.todo_color); + let future_style = Style::from(CONFIGURATION.future_color); + + printer.with_style( + if !printer.focused { + future_style + } else { + goal_reached_style + }, + |p| { + p.print( + (0, 0), + &format!("{:width$}", self.name(), width = CONFIGURATION.view_width), + ) + }, + ); + + for i in 1..=31 { + let day = NaiveDate::from_ymd_opt(year, month, i); + let day_style; + + if let Some(d) = day { + if self.reached_goal(d) { + day_style = goal_reached_style; + } else { + day_style = todo_style; + } + let coords: Vec2 = ((i % 7) * 3, i / 7 + 2).into(); + if let Some(c) = self.get_by_date(d) { + printer.with_style(day_style, |p| { + p.print(coords, &format!("{:^3}", c)); + }); + } else { + printer.with_style(future_style, |p| { + p.print(coords, &format!("{:^3}", CONFIGURATION.future_chr)); + }); + } + //printer.with_style(day_style, |p| { + // p.print(coords, &format!("{:^3}", c)); + // } else { + // p.print(coords, &format!("{:^3}", CONFIGURATION.future_chr)); + // } + //}); + } + } + } + fn required_size(&mut self, _: Vec2) -> Vec2 { + (25, 6).into() + } + + fn take_focus(&mut self, _: Direction) -> bool { + true + } + + fn on_event(&mut self, e: Event) -> EventResult { + let now = Local::now().naive_utc().date(); + match e { + Event::Key(Key::Enter) | Event::Char('n') => { + self.modify(now, TrackEvent::Increment); + return EventResult::Consumed(None); + } + Event::Key(Key::Backspace) | Event::Char('p') => { + self.modify(now, TrackEvent::Decrement); + return EventResult::Consumed(None); + } + _ => return EventResult::Ignored, + } + } +} + +macro_rules! auto_view_impl { + ($struct_name:ident) => { + impl View for $struct_name { + fn draw(&self, printer: &Printer) { + ShadowView::draw(self, printer); + } + fn required_size(&mut self, x: Vec2) -> Vec2 { + ShadowView::required_size(self, x) + } + + fn take_focus(&mut self, d: Direction) -> bool { + ShadowView::take_focus(self, d) + } + + fn on_event(&mut self, e: Event) -> EventResult { + ShadowView::on_event(self, e) + } + } + }; +} + +auto_view_impl!(Count); +auto_view_impl!(Bit); -- cgit v1.2.3