helix-mods/helix-term/src/ui/mod.rs

199 lines
6.1 KiB
Rust
Raw Normal View History

mod completion;
mod editor;
2021-03-05 08:07:46 +01:00
mod markdown;
2021-02-09 07:40:30 +01:00
mod menu;
mod picker;
2021-02-25 10:07:47 +01:00
mod popup;
mod prompt;
mod text;
pub use completion::Completion;
pub use editor::EditorView;
2021-03-05 08:07:46 +01:00
pub use markdown::Markdown;
2021-02-09 07:40:30 +01:00
pub use menu::Menu;
pub use picker::Picker;
2021-02-25 10:07:47 +01:00
pub use popup::Popup;
2020-12-15 11:29:56 +01:00
pub use prompt::{Prompt, PromptEvent};
pub use text::Text;
pub use tui::layout::Rect;
pub use tui::style::{Color, Modifier, Style};
2021-01-22 09:13:14 +01:00
use helix_core::regex::Regex;
use helix_core::register::Registers;
use helix_view::{Document, Editor, View};
2021-01-22 09:13:14 +01:00
2021-03-24 08:26:53 +01:00
use std::path::{Path, PathBuf};
2021-01-22 09:13:14 +01:00
pub fn regex_prompt(
cx: &mut crate::commands::Context,
prompt: String,
fun: impl Fn(&mut View, &mut Document, &mut Registers, Regex) + 'static,
2021-01-22 09:13:14 +01:00
) -> Prompt {
let view_id = cx.view().id;
let snapshot = cx.doc().selection(view_id).clone();
2021-01-22 09:13:14 +01:00
Prompt::new(
prompt,
|input: &str| Vec::new(), // this is fine because Vec::new() doesn't allocate
move |editor: &mut Editor, input: &str, event: PromptEvent| {
match event {
PromptEvent::Abort => {
// TODO: also revert text
let (view, doc) = editor.current();
doc.set_selection(view.id, snapshot.clone());
2021-01-22 09:13:14 +01:00
}
PromptEvent::Validate => {
// TODO: push_jump to store selection just before jump
2021-01-22 09:13:14 +01:00
}
PromptEvent::Update => {
// skip empty input, TODO: trigger default
if input.is_empty() {
return;
}
2021-01-22 09:13:14 +01:00
match Regex::new(input) {
Ok(regex) => {
let (view, doc, registers) = editor.current_with_registers();
2021-01-22 09:13:14 +01:00
// revert state to what it was before the last update
// TODO: also revert text
doc.set_selection(view.id, snapshot.clone());
2021-01-22 09:13:14 +01:00
fun(view, doc, registers, regex);
view.ensure_cursor_in_view(doc);
2021-01-22 09:13:14 +01:00
}
Err(_err) => (), // TODO: mark command line as error
}
}
}
},
)
}
pub fn file_picker(root: PathBuf) -> Picker<PathBuf> {
use ignore::Walk;
let files = Walk::new(root.clone()).filter_map(|entry| match entry {
Ok(entry) => {
// filter dirs, but we might need special handling for symlinks!
2021-04-01 04:01:11 +02:00
if !entry.file_type().map_or(false, |entry| entry.is_dir()) {
Some(entry.into_path())
} else {
None
}
}
Err(_err) => None,
});
2021-06-08 20:36:27 +02:00
let files = if root.join(".git").is_dir() {
files.collect()
} else {
const MAX: usize = 8192;
files.take(MAX).collect()
};
Picker::new(
2021-06-08 20:36:27 +02:00
files,
move |path: &PathBuf| {
// format_fn
2021-04-01 04:01:11 +02:00
path.strip_prefix(&root)
.unwrap_or(path)
.to_str()
.unwrap()
.into()
},
2021-03-29 08:21:48 +02:00
move |editor: &mut Editor, path: &PathBuf, action| {
let document_id = editor
2021-03-29 08:21:48 +02:00
.open(path.into(), action)
.expect("editor.open failed");
},
)
}
2020-12-21 08:23:05 +01:00
pub mod completers {
2021-03-22 05:16:56 +01:00
use crate::ui::prompt::Completion;
use std::borrow::Cow;
2021-05-07 10:19:45 +02:00
pub type Completer = fn(&str) -> Vec<Completion>;
// TODO: we could return an iter/lazy thing so it can fetch as many as it needs.
2021-03-22 05:16:56 +01:00
pub fn filename(input: &str) -> Vec<Completion> {
// Rust's filename handling is really annoying.
use ignore::WalkBuilder;
use std::path::{Path, PathBuf};
let path = Path::new(input);
let (dir, file_name) = if input.ends_with('/') {
(path.into(), None)
} else {
let file_name = path
.file_name()
.map(|file| file.to_str().unwrap().to_owned());
let path = match path.parent() {
Some(path) if !path.as_os_str().is_empty() => path.to_path_buf(),
// Path::new("h")'s parent is Some("")...
_ => std::env::current_dir().expect("couldn't determine current directory"),
};
(path, file_name)
};
let end = (input.len()..);
let mut files: Vec<_> = WalkBuilder::new(dir.clone())
.max_depth(Some(1))
.build()
.filter_map(|file| {
file.ok().map(|entry| {
2021-03-31 16:42:16 +02:00
let is_dir = entry.file_type().map_or(false, |entry| entry.is_dir());
2021-04-01 04:01:11 +02:00
let path = entry.path();
let mut path = path.strip_prefix(&dir).unwrap_or(path).to_path_buf();
if is_dir {
path.push("");
}
2021-04-01 04:01:11 +02:00
let path = path.to_str().unwrap().to_owned();
(end.clone(), Cow::from(path))
})
}) // TODO: unwrap or skip
.filter(|(_, path)| !path.is_empty()) // TODO
.collect();
// if empty, return a list of dirs and files in current dir
if let Some(file_name) = file_name {
use fuzzy_matcher::skim::SkimMatcherV2 as Matcher;
use fuzzy_matcher::FuzzyMatcher;
use std::cmp::Reverse;
let matcher = Matcher::default();
// inefficient, but we need to calculate the scores, filter out None, then sort.
let mut matches: Vec<_> = files
.into_iter()
.filter_map(|(range, file)| {
matcher
.fuzzy_match(&file, &file_name)
.map(|score| (file, score))
})
.collect();
let range = ((input.len() - file_name.len())..);
matches.sort_unstable_by_key(|(_file, score)| Reverse(*score));
files = matches
.into_iter()
.map(|(file, _)| (range.clone(), file))
.collect();
// TODO: complete to longest common match
}
files
}
}