262 lines
6 KiB
Rust
262 lines
6 KiB
Rust
use std::{
|
|
fs::{self, File},
|
|
io::{stdin, stdout, Stdout, Write},
|
|
ops::Range,
|
|
process::exit,
|
|
};
|
|
use termion::{
|
|
clear, cursor,
|
|
event::{Event, Key},
|
|
input::TermRead,
|
|
raw::{IntoRawMode, RawTerminal},
|
|
terminal_size,
|
|
};
|
|
|
|
const TAB_SIZE: usize = 4;
|
|
|
|
pub struct Editor {
|
|
text: String,
|
|
lines: Vec<Line>,
|
|
cursor: Cursor,
|
|
path: Option<String>,
|
|
term: RawTerminal<Stdout>,
|
|
quit: bool,
|
|
}
|
|
|
|
#[derive(Debug)]
|
|
struct Cursor {
|
|
line: usize,
|
|
column: usize,
|
|
// target_column: usize,
|
|
}
|
|
|
|
type Line = Range<usize>;
|
|
|
|
impl Editor {
|
|
pub fn new(path: Option<String>) -> Self {
|
|
let text = path
|
|
.as_ref()
|
|
.map(|path| {
|
|
fs::read_to_string(path).unwrap_or_else(|err| {
|
|
println!("Error: {err}");
|
|
exit(1);
|
|
})
|
|
})
|
|
.unwrap_or_default();
|
|
|
|
let term = stdout().into_raw_mode().unwrap();
|
|
|
|
Editor {
|
|
text,
|
|
lines: Vec::new(),
|
|
cursor: Cursor { line: 0, column: 0 },
|
|
term,
|
|
path,
|
|
quit: false,
|
|
}
|
|
}
|
|
|
|
pub fn run(mut self) {
|
|
print!("{}", clear::All);
|
|
stdout().flush().unwrap();
|
|
|
|
self.find_lines();
|
|
|
|
while !self.quit {
|
|
self.draw();
|
|
self.input();
|
|
}
|
|
print!("{}", clear::All);
|
|
}
|
|
|
|
fn input(&mut self) {
|
|
for event in stdin().events().take(1).flatten() {
|
|
// dbg!(&event);
|
|
if let Event::Key(key) = event {
|
|
match key {
|
|
Key::Esc => self.quit = true,
|
|
Key::Char(char) => self.insert_char(char),
|
|
Key::Backspace => self.backspace(),
|
|
Key::Delete => self.delete(),
|
|
Key::Left => self.move_left(),
|
|
Key::Right => self.move_right(),
|
|
Key::Up => self.move_up(),
|
|
Key::Down => self.move_down(),
|
|
Key::Ctrl('s') => self.save(),
|
|
_ => (),
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
fn move_left(&mut self) {
|
|
if self.cursor.column > 0 {
|
|
self.cursor.column = self.prev_char_index() - self.current_line().start;
|
|
} else if self.cursor.line > 0 {
|
|
self.cursor.line -= 1;
|
|
self.cursor.column = self.current_line().len();
|
|
}
|
|
}
|
|
|
|
fn move_right(&mut self) {
|
|
if self.cursor.column < self.current_line().len() {
|
|
self.cursor.column = self.next_char_index() - self.current_line().start;
|
|
} else if self.cursor.line < self.lines.len() - 1 {
|
|
self.cursor.line += 1;
|
|
self.cursor.column = 0;
|
|
}
|
|
}
|
|
|
|
fn move_up(&mut self) {
|
|
if self.cursor.line > 0 {
|
|
let physical_column = self.text
|
|
[self.current_line().start..(self.current_line().start + self.cursor.column)]
|
|
.chars()
|
|
.count();
|
|
self.cursor.line -= 1;
|
|
self.cursor.column = physical_column.min(self.current_line().len());
|
|
self.ensure_char_boundary();
|
|
}
|
|
}
|
|
|
|
fn move_down(&mut self) {
|
|
if self.cursor.line < self.lines.len() - 1 {
|
|
let physical_column = self.text
|
|
[self.current_line().start..(self.current_line().start + self.cursor.column)]
|
|
.chars()
|
|
.count();
|
|
self.cursor.line += 1;
|
|
self.cursor.column = physical_column.min(self.current_line().len());
|
|
self.ensure_char_boundary();
|
|
}
|
|
}
|
|
|
|
/// Moves cursor left until it is on a character (in case it was in the middle of a multi-byte character)
|
|
fn ensure_char_boundary(&mut self) {
|
|
while !self
|
|
.text
|
|
.is_char_boundary(self.current_line().start + self.cursor.column)
|
|
{
|
|
self.cursor.column -= 1;
|
|
}
|
|
}
|
|
|
|
fn current_line(&self) -> &Line {
|
|
self.lines.get(self.cursor.line).unwrap()
|
|
}
|
|
|
|
fn find_lines(&mut self) {
|
|
self.lines.clear();
|
|
let mut this_line = 0..0;
|
|
for (index, char) in self.text.char_indices() {
|
|
if char == '\n' {
|
|
this_line.end = index;
|
|
self.lines.push(this_line.clone());
|
|
this_line.start = index + 1;
|
|
}
|
|
}
|
|
this_line.end = self.text.len();
|
|
self.lines.push(this_line);
|
|
}
|
|
|
|
fn draw(&self) {
|
|
print!("{}", clear::All);
|
|
|
|
for (row, line) in self.lines.iter().enumerate() {
|
|
let text = &self.text[line.clone()];
|
|
print!(
|
|
"{}{}",
|
|
cursor::Goto(1, row as u16 + 1),
|
|
text.replace('\t', &" ".repeat(TAB_SIZE))
|
|
);
|
|
}
|
|
print!(
|
|
"{}({}, {})",
|
|
cursor::Goto(1, terminal_size().unwrap().1),
|
|
self.cursor.line,
|
|
self.cursor.column
|
|
);
|
|
|
|
print!(
|
|
"{}",
|
|
cursor::Goto(
|
|
self.physical_column() as u16 + 1,
|
|
self.cursor.line as u16 + 1
|
|
)
|
|
);
|
|
stdout().flush().unwrap();
|
|
}
|
|
|
|
fn insert_char(&mut self, ch: char) {
|
|
// eprintln!("inserting {ch} at {}", self.index());
|
|
self.text.insert(self.char_index(), ch);
|
|
self.find_lines();
|
|
self.move_right();
|
|
}
|
|
|
|
fn backspace(&mut self) {
|
|
if self.char_index() > 0 {
|
|
self.move_left();
|
|
self.text.remove(self.char_index());
|
|
self.find_lines();
|
|
}
|
|
}
|
|
|
|
fn delete(&mut self) {
|
|
if self.char_index() < self.text.len() {
|
|
self.text.remove(self.char_index());
|
|
self.find_lines();
|
|
}
|
|
}
|
|
|
|
/// Byte position of current character. May be text.len if cursor is at the end of the file
|
|
fn char_index(&self) -> usize {
|
|
self.current_line().start + self.cursor.column
|
|
}
|
|
|
|
/// Byte position of next character.
|
|
/// Returns text.len if cursor is on the last character
|
|
fn next_char_index(&self) -> usize {
|
|
self.text[self.char_index()..]
|
|
.char_indices()
|
|
.nth(1)
|
|
.map_or(self.text.len(), |(byte, _char)| byte + self.char_index())
|
|
}
|
|
|
|
/// Byte position of preceding character.
|
|
/// Panics if cursor is at index 0
|
|
fn prev_char_index(&self) -> usize {
|
|
self.text[..self.char_index()]
|
|
.char_indices()
|
|
.last()
|
|
.map(|(byte, _char)| byte)
|
|
.unwrap()
|
|
}
|
|
|
|
fn physical_column(&self) -> usize {
|
|
let start = self.current_line().start;
|
|
let end = self.char_index();
|
|
let preceding_chars = self.text[start..end].chars().count();
|
|
let preceding_tabs = self.text[start..end].chars().filter(|&c| c == '\t').count();
|
|
preceding_chars + preceding_tabs * (TAB_SIZE - 1)
|
|
}
|
|
|
|
fn save(&mut self) {
|
|
if self.path.is_none() {
|
|
self.path = Some(self.read_line("Save as: "));
|
|
}
|
|
let mut file = File::create(self.path.as_ref().unwrap()).unwrap();
|
|
file.write_all(self.text.as_bytes()).unwrap();
|
|
}
|
|
|
|
fn read_line(&self, prompt: &str) -> String {
|
|
// TODO: use events instead and allow cancelling with esc
|
|
self.term.suspend_raw_mode().unwrap();
|
|
print!("{}{prompt}", cursor::Goto(1, terminal_size().unwrap().1));
|
|
stdout().flush().unwrap();
|
|
let mut response = String::new();
|
|
stdin().read_line(&mut response).unwrap();
|
|
self.term.activate_raw_mode().unwrap();
|
|
response.trim_end().into()
|
|
}
|
|
}
|