use crate::explore_config::input::{
handle_editor_editing_input, handle_editor_normal_input, handle_search_input, handle_tree_input,
};
use crate::explore_config::types::{App, AppResult, EditorMode, Focus, NuValueType};
use crossterm::event::{
self, DisableMouseCapture, EnableMouseCapture, Event, KeyCode, KeyEventKind, KeyModifiers,
};
use crossterm::terminal::{
EnterAlternateScreen, LeaveAlternateScreen, disable_raw_mode, enable_raw_mode,
};
use nu_protocol::engine::{EngineState, Stack};
use ratatui::Terminal;
use ratatui::backend::CrosstermBackend;
use serde_json::Value;
use std::collections::HashMap;
use std::error::Error;
use std::io;
use std::sync::Arc;
pub fn run_config_tui(
json_data: Value,
output_file: Option<String>,
config_mode: bool,
nu_type_map: Option<HashMap<String, NuValueType>>,
doc_map: Option<HashMap<String, String>>,
engine_state: Arc<EngineState>,
stack: Arc<Stack>,
) -> Result<Option<Value>, Box<dyn Error>> {
enable_raw_mode()?;
let mut stdout = io::stdout();
crossterm::execute!(stdout, EnterAlternateScreen, EnableMouseCapture)?;
let backend = CrosstermBackend::new(stdout);
let mut terminal = Terminal::new(backend)?;
terminal.clear()?;
let mut app = App::new(
json_data,
output_file,
config_mode,
nu_type_map,
doc_map,
engine_state,
stack,
);
app.tree_state.select_first();
app.force_update_editor();
let res = run_config_app(&mut terminal, &mut app);
disable_raw_mode()?;
crossterm::execute!(
terminal.backend_mut(),
LeaveAlternateScreen,
DisableMouseCapture
)?;
terminal.show_cursor()?;
if let Err(err) = res {
eprintln!("Error: {err:?}");
return Ok(None);
}
if config_mode && app.modified {
Ok(Some(app.json_data))
} else {
Ok(None)
}
}
fn run_config_app(
terminal: &mut Terminal<CrosstermBackend<io::Stdout>>,
app: &mut App,
) -> io::Result<()> {
loop {
terminal.draw(|frame| app.draw(frame))?;
if let Event::Key(key) = event::read()? {
if key.kind != KeyEventKind::Press {
continue;
}
let in_editing_mode =
app.focus == Focus::Editor && app.editor_mode == EditorMode::Editing;
match key.code {
KeyCode::Char('c') if key.modifiers.contains(KeyModifiers::CONTROL) => {
return Ok(());
}
KeyCode::Char('s')
if key.modifiers.contains(KeyModifiers::CONTROL) && !in_editing_mode =>
{
if let Err(e) = app.save_to_file() {
app.status_message = format!("✗ Save failed: {}", e);
}
continue;
}
_ => {}
}
let result = match (app.focus, app.editor_mode) {
(Focus::Search, _) => handle_search_input(app, key.code, key.modifiers),
(Focus::Tree, _) => handle_tree_input(app, key.code, key.modifiers),
(Focus::Editor, EditorMode::Normal) => {
handle_editor_normal_input(app, key.code, key.modifiers)
}
(Focus::Editor, EditorMode::Editing) => {
handle_editor_editing_input(app, key.code, key.modifiers)
}
};
if let AppResult::Quit = result {
return Ok(());
}
}
}
}