use crate::app::App;
use ratatui::{
layout::{Constraint, Direction, Layout, Rect},
style::{Color, Modifier, Style},
text::{Line, Span},
widgets::{Block, Borders, Clear, Paragraph},
Frame,
};
pub fn render(_app: &App, frame: &mut Frame, area: Rect) {
let overlay = centered_rect(62, 85, area);
frame.render_widget(Clear, overlay);
let lines = vec![
Line::from(Span::styled(
"Keyboard Reference",
Style::default()
.add_modifier(Modifier::BOLD)
.fg(Color::Cyan),
)),
Line::from(""),
Line::from(Span::styled(
" Browsing (Intro / Examples)",
Style::default().fg(Color::Cyan).add_modifier(Modifier::BOLD),
)),
Line::from(""),
key_line("↑ / ↓", "Navigate module list"),
key_line("Tab", "Cycle views: Intro → Examples → Exercise → Free"),
key_line("PgUp / PgDn", "Scroll content"),
key_line("/", "Fuzzy search modules"),
key_line("d", "Cycle difficulty filter"),
key_line("P", "Progress summary"),
key_line("q", "Quit"),
Line::from(""),
Line::from(Span::styled(
" Exercise view",
Style::default().fg(Color::Cyan).add_modifier(Modifier::BOLD),
)),
Line::from(""),
key_line("Enter", "Submit command"),
key_line("↑ / ↓", "Browse command history"),
key_line("PgUp / PgDn", "Scroll output"),
key_line("← / →", "Move cursor"),
key_line("Ctrl+← / →", "Jump by word"),
key_line("Ctrl+N / Ctrl+P", "Next / previous exercise"),
key_line("Ctrl+T", "Reveal next hint"),
key_line("Ctrl+S", "Show/hide solution"),
key_line("Ctrl+F", "Toggle file viewer"),
key_line("Ctrl+R", "Reset exercise"),
key_line("Ctrl+L", "Clear output"),
key_line("Esc", "Back to browse"),
Line::from(""),
key_line("Ctrl+C", "Quit (anywhere)"),
key_line("P", "Progress summary (anywhere)"),
Line::from(""),
Line::from(Span::styled(
"Press Esc or ? to close",
Style::default().fg(Color::DarkGray),
)),
];
let para = Paragraph::new(lines).block(
Block::default()
.borders(Borders::ALL)
.title(" Help ")
.style(Style::default().bg(Color::Black)),
);
frame.render_widget(para, overlay);
}
fn key_line(key: &str, desc: &str) -> Line<'static> {
Line::from(vec![
Span::styled(
format!(" {:<16}", key),
Style::default()
.fg(Color::Yellow)
.add_modifier(Modifier::BOLD),
),
Span::raw(desc.to_string()),
])
}
fn centered_rect(percent_x: u16, percent_y: u16, r: Rect) -> Rect {
let vert = Layout::default()
.direction(Direction::Vertical)
.constraints([
Constraint::Percentage((100 - percent_y) / 2),
Constraint::Percentage(percent_y),
Constraint::Percentage((100 - percent_y) / 2),
])
.split(r);
Layout::default()
.direction(Direction::Horizontal)
.constraints([
Constraint::Percentage((100 - percent_x) / 2),
Constraint::Percentage(percent_x),
Constraint::Percentage((100 - percent_x) / 2),
])
.split(vert[1])[1]
}