use chrono::NaiveDate; use cursive::direction::Direction; use cursive::event::{Event, EventResult}; use cursive::{Printer, Vec2}; use typetag; use crate::habit::{Bit, Count, InnerData, TrackEvent}; use crate::views::ShadowView; pub trait Habit { type HabitType; fn get_by_date(&self, date: NaiveDate) -> Option<&Self::HabitType>; fn goal(&self) -> u32; fn insert_entry(&mut self, date: NaiveDate, val: Self::HabitType); fn modify(&mut self, date: NaiveDate, event: TrackEvent); fn name(&self) -> String; fn reached_goal(&self, date: NaiveDate) -> bool; fn remaining(&self, date: NaiveDate) -> u32; fn set_goal(&mut self, goal: Self::HabitType); fn set_name(&mut self, name: impl AsRef); fn inner_data_ref(&self) -> &InnerData; fn inner_data_mut_ref(&mut self) -> &mut InnerData; fn is_auto(&self) -> bool; } #[typetag::serde(tag = "type")] pub trait HabitWrapper: erased_serde::Serialize { fn draw(&self, printer: &Printer); fn goal(&self) -> u32; fn modify(&mut self, date: NaiveDate, event: TrackEvent); fn name(&self) -> String; fn on_event(&mut self, event: Event) -> EventResult; fn remaining(&self, date: NaiveDate) -> u32; fn required_size(&mut self, _: Vec2) -> Vec2; fn take_focus(&mut self, _: Direction) -> bool; fn inner_data_ref(&self) -> &InnerData; fn inner_data_mut_ref(&mut self) -> &mut InnerData; fn is_auto(&self) -> bool; } macro_rules! auto_habit_impl { ($struct_name:ident) => { #[typetag::serde] impl HabitWrapper for $struct_name { // ShadowView fn draw(&self, printer: &Printer) { ShadowView::draw(self, printer) } fn on_event(&mut self, event: Event) -> EventResult { ShadowView::on_event(self, event) } 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) } // Habit fn remaining(&self, date: NaiveDate) -> u32 { Habit::remaining(self, date) } fn goal(&self) -> u32 { Habit::goal(self) } fn modify(&mut self, date: NaiveDate, event: TrackEvent) { Habit::modify(self, date, event); } fn name(&self) -> String { Habit::name(self) } fn inner_data_ref(&self) -> &InnerData { Habit::inner_data_ref(self) } fn inner_data_mut_ref(&mut self) -> &mut InnerData { Habit::inner_data_mut_ref(self) } fn is_auto(&self) -> bool { Habit::is_auto(self) } } }; } auto_habit_impl!(Count); auto_habit_impl!(Bit);