mod engine;
use engine::Engine;
mod art;
mod format;
mod help;
pub mod highlight;
mod history;
pub mod options;
mod parser;
mod racer;
mod script;
use crossterm::event::KeyModifiers;
use crossterm::event::{Event, KeyCode, KeyEvent};
use highlight::theme::Theme;
use history::History;
use irust_api::{Command, GlobalVariables};
use irust_repl::Repl;
use options::Options;
use printer::{buffer::Buffer, printer::Printer};
use racer::Racer;
use script::Script;
pub type Result<T> = std::result::Result<T, Box<dyn std::error::Error>>;
pub struct IRust {
options: Options,
buffer: Buffer,
printer: Printer<std::io::Stdout>,
_engine: Engine,
exit_flag: bool,
theme: Theme,
repl: Repl,
global_variables: GlobalVariables,
history: History,
racer: Option<Racer>,
script_mg: Option<Box<dyn Script>>,
}
impl IRust {
pub fn new(options: Options) -> Self {
let repl = Repl::new_with_executor(options.toolchain, options.executor)
.expect("Could not create repl");
let mut global_variables = GlobalVariables::new();
let script_mg = Self::choose_script_mg(&options);
let prompt = script_mg
.as_ref()
.map(|script_mg| {
if let Some(prompt) = script_mg.input_prompt(&global_variables) {
prompt
} else {
options.input_prompt.clone()
}
})
.unwrap_or_else(|| options.input_prompt.clone());
global_variables.prompt_len = prompt.chars().count();
let printer = Printer::new(std::io::stdout(), prompt);
let racer = if options.enable_racer {
Racer::start()
} else {
None
};
let buffer = Buffer::new();
let _engine = Engine::default();
let exit_flag = false;
let theme = highlight::theme::theme().unwrap_or_default();
let history = History::new().unwrap_or_default();
IRust {
options,
buffer,
printer,
_engine,
exit_flag,
theme,
repl,
global_variables,
history,
racer,
script_mg,
}
}
fn prepare(&mut self) -> Result<()> {
self.printer.writer.raw.set_title(&format!(
"IRust: {}",
self.global_variables.get_cwd().display()
))?;
self.welcome()?;
self.printer.print_prompt_if_set()?;
Ok(())
}
pub fn print_input(&mut self) -> Result<()> {
let theme = &self.theme;
self.printer
.print_input(&|buffer| highlight::highlight(buffer, theme), &self.buffer)?;
Ok(())
}
pub fn run(&mut self) -> Result<()> {
self.prepare()?;
loop {
std::io::Write::flush(&mut self.printer.writer.raw)?;
match crossterm::event::read() {
Ok(ev) => {
self.handle_input_event(ev)?;
if self.exit_flag {
break Ok(());
}
}
Err(e) => break Err(format!("failed to read input. error: {}", e).into()),
}
}
}
fn handle_input_event(&mut self, ev: crossterm::event::Event) -> Result<()> {
self.update_script_state();
if let Some(command) = self.input_event_hook(ev) {
self.execute(command)?;
return Ok(());
}
match ev {
Event::Mouse(_) => (),
Event::Resize(width, height) => {
self.printer.cursor.update_dimensions(width, height);
self.execute(Command::HandleCtrlC)?;
}
Event::Key(key_event) => match key_event {
KeyEvent {
code: KeyCode::Char(c),
modifiers: KeyModifiers::NONE,
}
| KeyEvent {
code: KeyCode::Char(c),
modifiers: KeyModifiers::SHIFT,
} => self.execute(Command::HandleCharacter(c))?,
KeyEvent {
code: KeyCode::Char('e'),
modifiers: KeyModifiers::CONTROL,
} => self.execute(Command::HandleCtrlE)?,
KeyEvent {
code: KeyCode::Enter,
modifiers: KeyModifiers::ALT,
} => self.execute(Command::HandleAltEnter)?,
KeyEvent {
code: KeyCode::Enter,
..
} => self.execute(Command::HandleEnter(false))?,
KeyEvent {
code: KeyCode::Tab, ..
} => self.execute(Command::HandleTab)?,
KeyEvent {
code: KeyCode::BackTab,
..
} => self.execute(Command::HandleBackTab)?,
KeyEvent {
code: KeyCode::Left,
modifiers: KeyModifiers::NONE,
} => self.execute(Command::HandleLeft)?,
KeyEvent {
code: KeyCode::Right,
modifiers: KeyModifiers::NONE,
} => self.execute(Command::HandleRight)?,
KeyEvent {
code: KeyCode::Up, ..
} => self.execute(Command::HandleUp)?,
KeyEvent {
code: KeyCode::Down,
..
} => self.execute(Command::HandleDown)?,
KeyEvent {
code: KeyCode::Backspace,
..
} => self.execute(Command::HandleBackSpace)?,
KeyEvent {
code: KeyCode::Char('c'),
modifiers: KeyModifiers::CONTROL,
} => self.execute(Command::HandleCtrlC)?,
KeyEvent {
code: KeyCode::Char('d'),
modifiers: KeyModifiers::CONTROL,
} => {
self.execute(Command::HandleCtrlD)?;
}
KeyEvent {
code: KeyCode::Char('z'),
modifiers: KeyModifiers::CONTROL,
} => self.execute(Command::HandleCtrlZ)?,
KeyEvent {
code: KeyCode::Char('l'),
modifiers: KeyModifiers::CONTROL,
} => self.execute(Command::HandleCtrlL)?,
KeyEvent {
code: KeyCode::Char('r'),
modifiers: KeyModifiers::CONTROL,
} => self.execute(Command::HandleCtrlR)?,
KeyEvent {
code: KeyCode::Home,
..
} => self.execute(Command::HandleHome)?,
KeyEvent {
code: KeyCode::End, ..
} => self.execute(Command::HandleEnd)?,
KeyEvent {
code: KeyCode::Left,
modifiers: KeyModifiers::CONTROL,
} => self.execute(Command::HandleCtrlLeft)?,
KeyEvent {
code: KeyCode::Right,
modifiers: KeyModifiers::CONTROL,
} => self.execute(Command::HandleCtrlRight)?,
KeyEvent {
code: KeyCode::Delete,
..
} => {
self.execute(Command::HandleDelete)?;
self.execute(Command::PrintInput)?;
}
keyevent => {
if keyevent
.modifiers
.contains(KeyModifiers::CONTROL | KeyModifiers::ALT)
{
if let KeyCode::Char(c) = keyevent.code {
self.execute(Command::HandleCharacter(c))?;
}
}
}
},
}
Ok(())
}
}
impl Drop for IRust {
fn drop(&mut self) {
let _ = self.execute(Command::Exit);
if std::thread::panicking() {
let _ = self.printer.writer.raw.write("IRust panicked, to log the error you can redirect stderror to a file, example irust 2>log");
}
}
}