mirror of
https://github.com/danbulant/rush
synced 2026-05-19 04:18:35 +00:00
139 lines
3.7 KiB
Rust
139 lines
3.7 KiB
Rust
mod parser;
|
|
|
|
use std::io::{self, Read, Stdout, Write};
|
|
use std::cmp;
|
|
use std::convert::TryInto;
|
|
use std::ops::Add;
|
|
use std::process;
|
|
|
|
use termion::raw::{IntoRawMode, RawTerminal};
|
|
use termion::input::TermRead;
|
|
use termion::cursor::{self, DetectCursorPos};
|
|
use termion::event::*;
|
|
use termion::input::{MouseTerminal};
|
|
|
|
struct Term {
|
|
input: String,
|
|
idx: usize,
|
|
}
|
|
|
|
impl Term {
|
|
fn new() -> Term {
|
|
return Term {
|
|
input: String::new(),
|
|
idx: 0,
|
|
};
|
|
}
|
|
|
|
fn print(self: &Self, stdout: &mut RawTerminal<Stdout>) {
|
|
print!("{}", self.format(stdout));
|
|
}
|
|
fn format(self: &Self, stdout: &mut RawTerminal<Stdout>) -> String {
|
|
let (_, y) = stdout.cursor_pos().unwrap();
|
|
format!(
|
|
"{}{}{}{}{}",
|
|
termion::clear::CurrentLine,
|
|
termion::cursor::Goto(1, y),
|
|
&self.input,
|
|
termion::cursor::Left((self.input.len() - self.idx).try_into().unwrap()),
|
|
termion::cursor::Right(if self.input.len() > 0 { 1 } else { 0 } )
|
|
)
|
|
}
|
|
|
|
fn insert(self: &mut Self, idx: usize, char: char) {
|
|
self.input.insert(idx, char);
|
|
}
|
|
fn insert_str(self: &mut Self, idx: usize, str: &str) {
|
|
self.input.insert_str(idx, str);
|
|
}
|
|
fn remove(self: &mut Self, idx: usize) {
|
|
self.input.remove(idx);
|
|
}
|
|
}
|
|
|
|
struct Shell {
|
|
term: Term,
|
|
ctx: parser::vars::Context,
|
|
}
|
|
|
|
impl Shell {
|
|
fn new() -> Shell {
|
|
return Shell {
|
|
term: Term::new(),
|
|
ctx: parser::vars::Context {
|
|
scopes: Vec::new(),
|
|
parent_context: None,
|
|
},
|
|
};
|
|
}
|
|
}
|
|
|
|
fn main() {
|
|
loop {
|
|
let mut shell = editor();
|
|
shell.term.input += "\n";
|
|
parser::exec(&mut shell.term.input.as_bytes(), shell.ctx);
|
|
}
|
|
}
|
|
|
|
fn editor() -> Shell {
|
|
let stdin = io::stdin();
|
|
let mut stdout = io::stdout().into_raw_mode().unwrap();
|
|
let mut shell = Shell::new();
|
|
let mut should_execute = true;
|
|
for c in stdin.keys() {
|
|
let c = c.unwrap();
|
|
match c {
|
|
Key::Char('\n') => {
|
|
if shell.term.input.chars().nth(shell.term.idx).unwrap_or(' ') == '\\' {
|
|
shell.term.insert_str(shell.term.idx, "\\\n");
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
Key::Backspace => {
|
|
if shell.term.input.len() > 0 && shell.term.idx > 0 {
|
|
if shell.term.idx == shell.term.input.len() - 1 {
|
|
shell.term.input.pop();
|
|
} else {
|
|
shell.term.remove(shell.term.idx - 1);
|
|
}
|
|
shell.term.idx -= 1;
|
|
}
|
|
}
|
|
Key::Delete => {
|
|
if shell.term.idx < shell.term.input.len() {
|
|
shell.term.remove(shell.term.idx);
|
|
}
|
|
}
|
|
Key::End => {
|
|
shell.term.idx = cmp::max(shell.term.input.len(), 1) - 1;
|
|
}
|
|
Key::Home => {
|
|
shell.term.idx = 0;
|
|
}
|
|
Key::Left => {
|
|
if shell.term.idx > 0 {
|
|
shell.term.idx -= 1;
|
|
}
|
|
}
|
|
Key::Right => {
|
|
if shell.term.idx < shell.term.input.len() - 1 {
|
|
shell.term.idx += 1;
|
|
}
|
|
}
|
|
Key::Ctrl('c') => {
|
|
process::exit(1);
|
|
}
|
|
Key::Char(char) => {
|
|
shell.term.insert(shell.term.idx, char);
|
|
shell.term.idx += 1;
|
|
}
|
|
_ => {}
|
|
}
|
|
shell.term.print(&mut stdout);
|
|
stdout.flush().unwrap();
|
|
}
|
|
stdout.suspend_raw_mode();
|
|
shell
|
|
}
|