use envision::prelude::*;
struct TabsApp;
#[derive(Clone)]
struct State {
tabs: TabsState<String>,
}
#[derive(Clone, Debug)]
enum Msg {
Tabs(TabsMessage),
Quit,
}
impl App for TabsApp {
type State = State;
type Message = Msg;
fn init() -> (State, Command<Msg>) {
let mut tabs = TabsState::new(vec![
"Overview".to_string(),
"Details".to_string(),
"Settings".to_string(),
"Logs".to_string(),
]);
tabs.set_focused(true);
(State { tabs }, Command::none())
}
fn update(state: &mut State, msg: Msg) -> Command<Msg> {
match msg {
Msg::Tabs(m) => {
Tabs::<String>::update(&mut state.tabs, m);
}
Msg::Quit => return Command::quit(),
}
Command::none()
}
fn view(state: &State, frame: &mut Frame) {
let theme = Theme::default();
let area = frame.area();
let chunks = Layout::vertical([
Constraint::Length(3),
Constraint::Min(0),
Constraint::Length(1),
])
.split(area);
Tabs::<String>::view(
&state.tabs,
frame,
chunks[0],
&theme,
&ViewContext::default(),
);
let tab_name = state
.tabs
.selected_item()
.cloned()
.unwrap_or_else(|| "None".into());
let content_text = match tab_name.as_str() {
"Overview" => " System status: All services running",
"Details" => " CPU: 45% | Memory: 2.1 GB / 8 GB",
"Settings" => " Theme: Dark | Language: English",
"Logs" => " [INFO] Application started successfully",
_ => " Unknown tab",
};
let content = ratatui::widgets::Paragraph::new(content_text).block(
ratatui::widgets::Block::default()
.borders(ratatui::widgets::Borders::ALL)
.title(tab_name),
);
frame.render_widget(content, chunks[1]);
let status = " Left/Right: switch tabs, q: quit";
frame.render_widget(
ratatui::widgets::Paragraph::new(status).style(Style::default().fg(Color::DarkGray)),
chunks[2],
);
}
fn handle_event_with_state(state: &State, event: &Event) -> Option<Msg> {
if let Some(key) = event.as_key() {
if matches!(key.code, KeyCode::Char('q') | KeyCode::Esc) {
return Some(Msg::Quit);
}
}
state.tabs.handle_event(event).map(Msg::Tabs)
}
}
fn main() -> Result<(), Box<dyn std::error::Error>> {
let mut vt = Runtime::<TabsApp, _>::virtual_terminal(60, 10)?;
println!("=== Tabs Example ===\n");
vt.tick()?;
println!("Initial tabs (Overview selected):");
println!("{}\n", vt.display());
vt.dispatch(Msg::Tabs(TabsMessage::Right));
vt.tick()?;
println!("After switching to Details:");
println!("{}\n", vt.display());
vt.dispatch(Msg::Tabs(TabsMessage::Right));
vt.tick()?;
println!("After switching to Settings:");
println!("{}\n", vt.display());
Ok(())
}