refactor: move clock to module in preparation for cli
cargo devel CI / cargo CI (push) Has been cancelled Details

This commit is contained in:
Christoph J. Scherr 2024-07-09 20:08:58 +02:00
parent 652eb61f92
commit 572db7f182
3 changed files with 121 additions and 85 deletions

View File

@ -14,6 +14,7 @@ categories = ["date-and-time"]
[dependencies] [dependencies]
anyhow = "1.0.86"
chrono = "0.4.38" chrono = "0.4.38"
libpt = { version = "0.6.0", features = ["cli"] } libpt = { version = "0.6.0", features = ["cli"] }
ratatui = "0.27.0" ratatui = "0.27.0"

105
src/clock.rs Normal file
View File

@ -0,0 +1,105 @@
#![warn(clippy::pedantic, clippy::style, clippy::nursery)]
#![allow(clippy::question_mark_used)]
use clap::Parser;
use libpt::cli::{args::VerbosityLevel, args::HELP_TEMPLATE, clap};
use chrono::SubsecRound;
use ratatui::crossterm::event::{
self, poll, DisableMouseCapture, EnableMouseCapture, Event, KeyCode, KeyModifiers,
};
use ratatui::layout::Alignment;
use ratatui::widgets::{Block, Padding};
use ratatui::{
backend::CrosstermBackend,
crossterm::{
execute,
terminal::{disable_raw_mode, enable_raw_mode, EnterAlternateScreen, LeaveAlternateScreen},
},
layout::{Constraint, Direction, Layout, Rect},
style::{Style, Stylize},
widgets::Paragraph,
Terminal,
};
use std::io::Stdout;
use std::time::Duration;
/// Make your terminal into a big clock
#[derive(Parser, Debug, Clone, PartialEq, Eq, Hash)]
#[command(help_template = HELP_TEMPLATE)]
pub(crate) struct Clock {
#[command(flatten)]
pub verbose: VerbosityLevel,
}
impl Clock {
pub(crate) fn run(
self,
terminal: &mut Terminal<CrosstermBackend<Stdout>>,
) -> anyhow::Result<()> {
loop {
let raw_time = chrono::Local::now().round_subsecs(0);
let splits: Vec<String> = raw_time
.naive_local()
.to_string()
.split_whitespace()
.map(str::to_string)
.collect();
let fdate: String = splits[0].clone();
let ftime: String = splits[1].clone();
terminal.draw(|frame| {
let root = frame.size();
let space = Block::bordered()
.padding(Padding::new(
root.width / 8,
root.width / 8,
root.height / 8,
root.height / 8,
))
.title(env!("CARGO_PKG_NAME"))
.title_bottom(env!("CARGO_PKG_VERSION"))
.title_alignment(Alignment::Center)
.title_style(Style::new().bold());
let a = space.inner(root);
let parts = Self::partition(a);
let timew = tui_big_text::BigText::builder()
.style(Style::new().red())
.lines(vec![ftime.into()])
.alignment(Alignment::Center)
.build()
.expect("could not render time widget");
let datew = Paragraph::new(fdate)
.blue()
.alignment(Alignment::Left)
.block(Block::new().padding(Padding::new(
parts.0.left(),
parts.0.right() / 3,
0,
0,
)));
frame.render_widget(space, root);
frame.render_widget(timew, parts.1);
frame.render_widget(datew, parts.0);
})?;
if poll(Duration::from_millis(100))? {
if let Event::Key(key) = event::read()? {
if key.code == KeyCode::Char('q')
|| key.code == KeyCode::Esc
|| (key.modifiers.contains(KeyModifiers::CONTROL)
&& key.code == KeyCode::Char('c'))
{
break;
}
}
}
}
Ok(())
}
fn partition(r: Rect) -> (Rect, Rect) {
let part = Layout::default()
.direction(Direction::Vertical)
.constraints([Constraint::Percentage(13), Constraint::Min(0)])
.split(r);
(part[0], part[1])
}
}

View File

@ -1,26 +1,19 @@
#![warn(clippy::pedantic, clippy::style, clippy::nursery)] use std::io;
#![allow(clippy::question_mark_used)]
use chrono::SubsecRound; use libpt::cli::clap::Parser;
use ratatui::crossterm::event::{ use ratatui::backend::CrosstermBackend;
self, poll, DisableMouseCapture, EnableMouseCapture, Event, KeyCode, KeyModifiers, use ratatui::crossterm::event::{DisableMouseCapture, EnableMouseCapture};
use ratatui::crossterm::execute;
use ratatui::crossterm::terminal::{
disable_raw_mode, enable_raw_mode, EnterAlternateScreen, LeaveAlternateScreen,
}; };
use ratatui::layout::Alignment; use ratatui::Terminal;
use ratatui::widgets::{Block, Padding};
use ratatui::{
backend::CrosstermBackend,
crossterm::{
execute,
terminal::{disable_raw_mode, enable_raw_mode, EnterAlternateScreen, LeaveAlternateScreen},
},
layout::{Constraint, Direction, Layout, Rect},
style::{Style, Stylize},
widgets::Paragraph,
Terminal,
};
use std::{io, time::Duration};
fn main() -> Result<(), io::Error> { use self::clock::Clock;
mod clock;
fn main() -> anyhow::Result<()> {
// setup terminal // setup terminal
enable_raw_mode()?; enable_raw_mode()?;
let mut stdout = io::stdout(); let mut stdout = io::stdout();
@ -28,62 +21,8 @@ fn main() -> Result<(), io::Error> {
let backend = CrosstermBackend::new(stdout); let backend = CrosstermBackend::new(stdout);
let mut terminal = Terminal::new(backend)?; let mut terminal = Terminal::new(backend)?;
loop { let clock = Clock::parse();
let raw_time = chrono::Local::now().round_subsecs(0); clock.run(&mut terminal)?;
let splits: Vec<String> = raw_time
.naive_local()
.to_string()
.split_whitespace()
.map(str::to_string)
.collect();
let fdate: String = splits[0].clone();
let ftime: String = splits[1].clone();
terminal.draw(|frame| {
let root = frame.size();
let space = Block::bordered()
.padding(Padding::new(
root.width / 8,
root.width / 8,
root.height / 8,
root.height / 8,
))
.title(env!("CARGO_PKG_NAME"))
.title_bottom(env!("CARGO_PKG_VERSION"))
.title_alignment(Alignment::Center)
.title_style(Style::new().bold());
let a = space.inner(root);
let parts = partition(a);
let timew = tui_big_text::BigText::builder()
.style(Style::new().red())
.lines(vec![ftime.into()])
.alignment(Alignment::Center)
.build()
.expect("could not render time widget");
let datew = Paragraph::new(fdate)
.blue()
.alignment(Alignment::Left)
.block(Block::new().padding(Padding::new(
parts.0.left(),
parts.0.right() / 3,
0,
0,
)));
frame.render_widget(space, root);
frame.render_widget(timew, parts.1);
frame.render_widget(datew, parts.0);
})?;
if poll(Duration::from_millis(100))? {
if let Event::Key(key) = event::read()? {
if key.code == KeyCode::Char('q')
|| key.code == KeyCode::Esc
|| (key.modifiers.contains(KeyModifiers::CONTROL)
&& key.code == KeyCode::Char('c'))
{
break;
}
}
}
}
// restore terminal // restore terminal
disable_raw_mode()?; disable_raw_mode()?;
@ -96,12 +35,3 @@ fn main() -> Result<(), io::Error> {
Ok(()) Ok(())
} }
fn partition(r: Rect) -> (Rect, Rect) {
let part = Layout::default()
.direction(Direction::Vertical)
.constraints([Constraint::Percentage(13), Constraint::Min(0)])
.split(r);
(part[0], part[1])
}