extern crate log; extern crate stderrlog; extern crate termion; extern crate tui; use std::io; use std::sync::mpsc; use std::thread; use termion::event; use termion::input::TermRead; use tui::backend::MouseBackend; use tui::layout::{Constraint, Direction, Layout, Rect}; use tui::widgets::{Block, Borders, Widget}; use tui::Terminal; struct App { size: Rect, } impl App { fn new() -> App { App { size: Rect::default(), } } } enum Event { Input(event::Key), } fn main() { stderrlog::new().verbosity(4).init().unwrap(); // Terminal initialization let backend = MouseBackend::new().unwrap(); let mut terminal = Terminal::new(backend).unwrap(); // Channels let (tx, rx) = mpsc::channel(); let input_tx = tx.clone(); // Input thread::spawn(move || { let stdin = io::stdin(); for c in stdin.keys() { let evt = c.unwrap(); input_tx.send(Event::Input(evt)).unwrap(); if evt == event::Key::Char('q') { break; } } }); // App let mut app = App::new(); // First draw call terminal.clear().unwrap(); terminal.hide_cursor().unwrap(); app.size = terminal.size().unwrap(); draw(&mut terminal, &app).unwrap(); loop { let size = terminal.size().unwrap(); if size != app.size { terminal.resize(size).unwrap(); app.size = size; } let evt = rx.recv().unwrap(); match evt { Event::Input(input) => if let event::Key::Char('q') = input { break; }, } draw(&mut terminal, &app).unwrap(); } terminal.show_cursor().unwrap(); } fn draw(t: &mut Terminal, app: &App) -> Result<(), io::Error> { t.draw(|mut f| { let chunks = Layout::default() .direction(Direction::Vertical) .constraints( [ Constraint::Percentage(10), Constraint::Percentage(80), Constraint::Percentage(10), ].as_ref(), ) .split(app.size); Block::default() .title("Block") .borders(Borders::ALL) .render(&mut f, chunks[0]); Block::default() .title("Block 2") .borders(Borders::ALL) .render(&mut f, chunks[2]); }) }