use rustyline::completion::{Completer, Pair};
use rustyline::error::ReadlineError;
use rustyline::highlight::Highlighter;
use rustyline::hint::Hinter;
use rustyline::validate::Validator;
use rustyline::{Config, Editor, Helper};
use std::borrow::Cow;
use crate::config;
#[derive(Clone)]
struct AbstractHelper {
commands: Vec<String>,
}
impl AbstractHelper {
fn new() -> Self {
let commands = vec![
"/help", "/clear", "/compact", "/cost", "/commit", "/review", "/memory", "/model",
"/config", "/diff", "/resume", "/exit", "/quit",
]
.into_iter()
.map(String::from)
.collect();
Self { commands }
}
}
impl Completer for AbstractHelper {
type Candidate = Pair;
fn complete(
&self,
line: &str,
pos: usize,
_ctx: &rustyline::Context<'_>,
) -> rustyline::Result<(usize, Vec<Pair>)> {
if line.starts_with('/') {
let candidates: Vec<Pair> = self
.commands
.iter()
.filter(|c| c.starts_with(line))
.map(|c| Pair {
display: c.clone(),
replacement: c.clone(),
})
.collect();
return Ok((0, candidates));
}
Ok((pos, vec![]))
}
}
impl Hinter for AbstractHelper {
type Hint = String;
fn hint(&self, _line: &str, _pos: usize, _ctx: &rustyline::Context<'_>) -> Option<String> {
None
}
}
impl Highlighter for AbstractHelper {
fn highlight_prompt<'b, 's: 'b, 'p: 'b>(
&'s self,
prompt: &'p str,
_default: bool,
) -> Cow<'b, str> {
Cow::Borrowed(prompt)
}
}
impl Validator for AbstractHelper {}
impl Helper for AbstractHelper {}
pub struct InputReader {
editor: Editor<AbstractHelper, rustyline::history::DefaultHistory>,
}
impl InputReader {
pub fn new() -> anyhow::Result<Self> {
let config = Config::builder()
.auto_add_history(true)
.max_history_size(1000)?
.build();
let mut editor = Editor::with_config(config)?;
editor.set_helper(Some(AbstractHelper::new()));
let history_path = config::history_path();
if history_path.exists() {
let _ = editor.load_history(&history_path);
}
Ok(Self { editor })
}
pub fn readline(&mut self, prompt: &str) -> Option<String> {
match self.editor.readline(prompt) {
Ok(line) => {
let trimmed = line.trim().to_string();
if trimmed.is_empty() {
return Some(String::new());
}
Some(trimmed)
}
Err(ReadlineError::Eof) | Err(ReadlineError::Interrupted) => None,
Err(_) => None,
}
}
pub fn save_history(&mut self) {
let history_path = config::history_path();
if let Some(parent) = history_path.parent() {
let _ = std::fs::create_dir_all(parent);
}
let _ = self.editor.save_history(&history_path);
}
}