picker: Factor out file picker, we want to reuse code for other pickers.
This commit is contained in:
parent
0b63e838e0
commit
25aa45e76c
3 changed files with 71 additions and 43 deletions
|
@ -458,7 +458,7 @@ pub fn command_mode(cx: &mut Context) {
|
||||||
}
|
}
|
||||||
pub fn file_picker(cx: &mut Context) {
|
pub fn file_picker(cx: &mut Context) {
|
||||||
cx.callback = Some(Box::new(|compositor: &mut Compositor| {
|
cx.callback = Some(Box::new(|compositor: &mut Compositor| {
|
||||||
let picker = ui::Picker::new();
|
let picker = ui::file_picker("./");
|
||||||
compositor.push(Box::new(picker));
|
compositor.push(Box::new(picker));
|
||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
|
|
|
@ -14,3 +14,35 @@ pub use tui::style::{Color, Modifier, Style};
|
||||||
pub fn text_color() -> Style {
|
pub fn text_color() -> Style {
|
||||||
Style::default().fg(Color::Rgb(219, 191, 239)) // lilac
|
Style::default().fg(Color::Rgb(219, 191, 239)) // lilac
|
||||||
}
|
}
|
||||||
|
|
||||||
|
use std::path::PathBuf;
|
||||||
|
pub fn file_picker(root: &str) -> Picker<PathBuf> {
|
||||||
|
use ignore::Walk;
|
||||||
|
// TODO: determine root based on git root
|
||||||
|
let files = Walk::new(root).filter_map(|entry| match entry {
|
||||||
|
Ok(entry) => {
|
||||||
|
// filter dirs, but we might need special handling for symlinks!
|
||||||
|
if !entry.file_type().unwrap().is_dir() {
|
||||||
|
Some(entry.into_path())
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(_err) => None,
|
||||||
|
});
|
||||||
|
|
||||||
|
const MAX: usize = 1024;
|
||||||
|
|
||||||
|
use helix_view::Editor;
|
||||||
|
Picker::new(
|
||||||
|
files.take(MAX).collect(),
|
||||||
|
|path: &PathBuf| {
|
||||||
|
// format_fn
|
||||||
|
path.strip_prefix("./").unwrap().to_str().unwrap() // TODO: render paths without ./
|
||||||
|
},
|
||||||
|
|editor: &mut Editor, path: &PathBuf| {
|
||||||
|
let size = editor.view().unwrap().size;
|
||||||
|
editor.open(path.into(), size);
|
||||||
|
},
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
|
@ -9,16 +9,13 @@ use tui::{
|
||||||
|
|
||||||
use fuzzy_matcher::skim::SkimMatcherV2 as Matcher;
|
use fuzzy_matcher::skim::SkimMatcherV2 as Matcher;
|
||||||
use fuzzy_matcher::FuzzyMatcher;
|
use fuzzy_matcher::FuzzyMatcher;
|
||||||
use ignore::Walk;
|
|
||||||
|
|
||||||
use std::path::PathBuf;
|
|
||||||
|
|
||||||
use crate::ui::{Prompt, PromptEvent};
|
use crate::ui::{Prompt, PromptEvent};
|
||||||
use helix_core::Position;
|
use helix_core::Position;
|
||||||
use helix_view::Editor;
|
use helix_view::Editor;
|
||||||
|
|
||||||
pub struct Picker {
|
pub struct Picker<T> {
|
||||||
files: Vec<PathBuf>,
|
options: Vec<T>,
|
||||||
// filter: String,
|
// filter: String,
|
||||||
matcher: Box<Matcher>,
|
matcher: Box<Matcher>,
|
||||||
/// (index, score)
|
/// (index, score)
|
||||||
|
@ -27,22 +24,17 @@ pub struct Picker {
|
||||||
cursor: usize,
|
cursor: usize,
|
||||||
// pattern: String,
|
// pattern: String,
|
||||||
prompt: Prompt,
|
prompt: Prompt,
|
||||||
|
|
||||||
|
format_fn: Box<dyn Fn(&T) -> &str>,
|
||||||
|
callback_fn: Box<dyn Fn(&mut Editor, &T)>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Picker {
|
impl<T> Picker<T> {
|
||||||
pub fn new() -> Self {
|
pub fn new(
|
||||||
let files = Walk::new("./").filter_map(|entry| match entry {
|
options: Vec<T>,
|
||||||
Ok(entry) => {
|
format_fn: impl Fn(&T) -> &str + 'static,
|
||||||
// filter dirs, but we might need special handling for symlinks!
|
callback_fn: impl Fn(&mut Editor, &T) + 'static,
|
||||||
if !entry.file_type().unwrap().is_dir() {
|
) -> Self {
|
||||||
Some(entry.into_path())
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Err(_err) => None,
|
|
||||||
});
|
|
||||||
|
|
||||||
let prompt = Prompt::new(
|
let prompt = Prompt::new(
|
||||||
"".to_string(),
|
"".to_string(),
|
||||||
|pattern: &str| Vec::new(),
|
|pattern: &str| Vec::new(),
|
||||||
|
@ -51,14 +43,14 @@ impl Picker {
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
const MAX: usize = 1024;
|
|
||||||
|
|
||||||
let mut picker = Self {
|
let mut picker = Self {
|
||||||
files: files.take(MAX).collect(),
|
options,
|
||||||
matcher: Box::new(Matcher::default()),
|
matcher: Box::new(Matcher::default()),
|
||||||
matches: Vec::new(),
|
matches: Vec::new(),
|
||||||
cursor: 0,
|
cursor: 0,
|
||||||
prompt,
|
prompt,
|
||||||
|
format_fn: Box::new(format_fn),
|
||||||
|
callback_fn: Box::new(callback_fn),
|
||||||
};
|
};
|
||||||
|
|
||||||
// TODO: scoring on empty input should just use a fastpath
|
// TODO: scoring on empty input should just use a fastpath
|
||||||
|
@ -70,9 +62,10 @@ impl Picker {
|
||||||
pub fn score(&mut self) {
|
pub fn score(&mut self) {
|
||||||
// need to borrow via pattern match otherwise it complains about simultaneous borrow
|
// need to borrow via pattern match otherwise it complains about simultaneous borrow
|
||||||
let Self {
|
let Self {
|
||||||
ref mut files,
|
ref mut options,
|
||||||
ref mut matcher,
|
ref mut matcher,
|
||||||
ref mut matches,
|
ref mut matches,
|
||||||
|
ref format_fn,
|
||||||
..
|
..
|
||||||
} = *self;
|
} = *self;
|
||||||
|
|
||||||
|
@ -80,15 +73,19 @@ impl Picker {
|
||||||
|
|
||||||
// reuse the matches allocation
|
// reuse the matches allocation
|
||||||
matches.clear();
|
matches.clear();
|
||||||
matches.extend(self.files.iter().enumerate().filter_map(|(index, path)| {
|
matches.extend(
|
||||||
match path.to_str() {
|
self.options
|
||||||
// TODO: using fuzzy_indices could give us the char idx for match highlighting
|
.iter()
|
||||||
Some(path) => matcher
|
.enumerate()
|
||||||
.fuzzy_match(path, pattern)
|
.filter_map(|(index, option)| {
|
||||||
.map(|score| (index, score)),
|
// TODO: maybe using format_fn isn't the best idea here
|
||||||
None => None,
|
let text = (format_fn)(option);
|
||||||
}
|
// TODO: using fuzzy_indices could give us the char idx for match highlighting
|
||||||
}));
|
matcher
|
||||||
|
.fuzzy_match(text, pattern)
|
||||||
|
.map(|score| (index, score))
|
||||||
|
}),
|
||||||
|
);
|
||||||
matches.sort_unstable_by_key(|(_, score)| -score);
|
matches.sort_unstable_by_key(|(_, score)| -score);
|
||||||
|
|
||||||
// reset cursor position
|
// reset cursor position
|
||||||
|
@ -101,15 +98,15 @@ impl Picker {
|
||||||
|
|
||||||
pub fn move_down(&mut self) {
|
pub fn move_down(&mut self) {
|
||||||
// TODO: len - 1
|
// TODO: len - 1
|
||||||
if self.cursor < self.files.len() {
|
if self.cursor < self.options.len() {
|
||||||
self.cursor += 1;
|
self.cursor += 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn selection(&self) -> Option<&PathBuf> {
|
pub fn selection(&self) -> Option<&T> {
|
||||||
self.matches
|
self.matches
|
||||||
.get(self.cursor)
|
.get(self.cursor)
|
||||||
.map(|(index, _score)| &self.files[*index])
|
.map(|(index, _score)| &self.options[*index])
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -118,7 +115,7 @@ impl Picker {
|
||||||
// - on input change:
|
// - on input change:
|
||||||
// - score all the names in relation to input
|
// - score all the names in relation to input
|
||||||
|
|
||||||
impl Component for Picker {
|
impl<T> Component for Picker<T> {
|
||||||
fn handle_event(&mut self, event: Event, cx: &mut Context) -> EventResult {
|
fn handle_event(&mut self, event: Event, cx: &mut Context) -> EventResult {
|
||||||
let key_event = match event {
|
let key_event = match event {
|
||||||
Event::Key(event) => event,
|
Event::Key(event) => event,
|
||||||
|
@ -163,9 +160,8 @@ impl Component for Picker {
|
||||||
code: KeyCode::Enter,
|
code: KeyCode::Enter,
|
||||||
..
|
..
|
||||||
} => {
|
} => {
|
||||||
let size = cx.editor.view().unwrap().size;
|
if let Some(option) = self.selection() {
|
||||||
if let Some(path) = self.selection() {
|
(self.callback_fn)(&mut cx.editor, option);
|
||||||
cx.editor.open(path.into(), size);
|
|
||||||
}
|
}
|
||||||
return close_fn;
|
return close_fn;
|
||||||
}
|
}
|
||||||
|
@ -238,10 +234,10 @@ impl Component for Picker {
|
||||||
let rows = inner.height - 2; // -1 for search bar
|
let rows = inner.height - 2; // -1 for search bar
|
||||||
|
|
||||||
let files = self.matches.iter().map(|(index, _score)| {
|
let files = self.matches.iter().map(|(index, _score)| {
|
||||||
(index, self.files.get(*index).unwrap()) // get_unchecked
|
(index, self.options.get(*index).unwrap()) // get_unchecked
|
||||||
});
|
});
|
||||||
|
|
||||||
for (i, (_index, file)) in files.take(rows as usize).enumerate() {
|
for (i, (_index, option)) in files.take(rows as usize).enumerate() {
|
||||||
if i == self.cursor {
|
if i == self.cursor {
|
||||||
surface.set_string(inner.x + 1, inner.y + 2 + i as u16, ">", selected);
|
surface.set_string(inner.x + 1, inner.y + 2 + i as u16, ">", selected);
|
||||||
}
|
}
|
||||||
|
@ -249,7 +245,7 @@ impl Component for Picker {
|
||||||
surface.set_stringn(
|
surface.set_stringn(
|
||||||
inner.x + 3,
|
inner.x + 3,
|
||||||
inner.y + 2 + i as u16,
|
inner.y + 2 + i as u16,
|
||||||
file.strip_prefix("./").unwrap().to_str().unwrap(), // TODO: render paths without ./
|
(self.format_fn)(option),
|
||||||
inner.width as usize - 1,
|
inner.width as usize - 1,
|
||||||
if i == self.cursor { selected } else { style },
|
if i == self.cursor { selected } else { style },
|
||||||
);
|
);
|
||||||
|
|
Loading…
Add table
Reference in a new issue