This commit is contained in:
Shaun_Sheep 2025-04-01 18:06:40 +00:00 committed by GitHub
commit 8d01df93d6
No known key found for this signature in database
GPG key ID: B5690EEEBB952194

View file

@ -268,6 +268,10 @@ pub struct Picker<T: 'static + Send + Sync, D: 'static> {
/// An event handler for syntax highlighting the currently previewed file. /// An event handler for syntax highlighting the currently previewed file.
preview_highlight_handler: Sender<Arc<Path>>, preview_highlight_handler: Sender<Arc<Path>>,
dynamic_query_handler: Option<Sender<DynamicQueryChange>>, dynamic_query_handler: Option<Sender<DynamicQueryChange>>,
preview_scroll_offset: (Direction, usize),
preview_height: u16,
cursor_picker: u32,
} }
impl<T: 'static + Send + Sync, D: 'static + Send + Sync> Picker<T, D> { impl<T: 'static + Send + Sync, D: 'static + Send + Sync> Picker<T, D> {
@ -389,6 +393,9 @@ impl<T: 'static + Send + Sync, D: 'static + Send + Sync> Picker<T, D> {
file_fn: None, file_fn: None,
preview_highlight_handler: PreviewHighlightHandler::<T, D>::default().spawn(), preview_highlight_handler: PreviewHighlightHandler::<T, D>::default().spawn(),
dynamic_query_handler: None, dynamic_query_handler: None,
preview_scroll_offset: (Direction::Forward, 0),
preview_height: 0,
cursor_picker: 0,
} }
} }
@ -440,6 +447,44 @@ impl<T: 'static + Send + Sync, D: 'static + Send + Sync> Picker<T, D> {
self self
} }
/// Moves the picker file preview by a number of lines, either down (`Forward`) or up (`Backward`)
fn move_preview_by(&mut self, amount: usize, move_direction: Direction) {
let (current_scroll_direction, current_scroll_offset) = self.preview_scroll_offset;
match move_direction {
Direction::Backward => match current_scroll_direction {
Direction::Backward => {
self.preview_scroll_offset.1 = current_scroll_offset.saturating_add(amount);
}
Direction::Forward => {
if let Some(change) = current_scroll_offset.checked_sub(amount) {
self.preview_scroll_offset.1 = change;
} else {
self.preview_scroll_offset = (
Direction::Backward,
amount.saturating_sub(current_scroll_offset),
);
}
}
},
Direction::Forward => match current_scroll_direction {
Direction::Backward => {
if let Some(change) = current_scroll_offset.checked_sub(amount) {
self.preview_scroll_offset.1 = change;
} else {
self.preview_scroll_offset = (
Direction::Forward,
amount.saturating_sub(current_scroll_offset),
);
}
}
Direction::Forward => {
self.preview_scroll_offset.1 = current_scroll_offset.saturating_add(amount);
}
},
};
}
/// Move the cursor by a number of lines, either down (`Forward`) or up (`Backward`) /// Move the cursor by a number of lines, either down (`Forward`) or up (`Backward`)
pub fn move_by(&mut self, amount: u32, direction: Direction) { pub fn move_by(&mut self, amount: u32, direction: Direction) {
let len = self.matcher.snapshot().matched_item_count(); let len = self.matcher.snapshot().matched_item_count();
@ -872,6 +917,15 @@ impl<T: 'static + Send + Sync, D: 'static + Send + Sync> Picker<T, D> {
let inner = inner.inner(margin); let inner = inner.inner(margin);
BLOCK.render(area, surface); BLOCK.render(area, surface);
let mut preview_scroll_offset = self.preview_scroll_offset;
// Reset preview scroll if cursor moved
let cursor_position = self.cursor_picker;
if self.cursor != cursor_position {
preview_scroll_offset = (Direction::Forward, 0);
self.cursor_picker = self.cursor;
}
if let Some((preview, range)) = self.get_preview(cx.editor) { if let Some((preview, range)) = self.get_preview(cx.editor) {
let doc = match preview.document() { let doc = match preview.document() {
Some(doc) Some(doc)
@ -905,6 +959,7 @@ impl<T: 'static + Send + Sync, D: 'static + Send + Sync> Picker<T, D> {
return; return;
} }
}; };
let doc_height = doc.text().len_lines();
let mut offset = ViewPosition::default(); let mut offset = ViewPosition::default();
if let Some((start_line, end_line)) = range { if let Some((start_line, end_line)) = range {
@ -933,6 +988,28 @@ impl<T: 'static + Send + Sync, D: 'static + Send + Sync> Picker<T, D> {
} }
} }
let mut current_line = doc.text().slice(..).char_to_line(offset.anchor);
preview_scroll_offset.1 = match preview_scroll_offset.0 {
Direction::Backward => preview_scroll_offset.1.min(current_line),
Direction::Forward => preview_scroll_offset.1.min(
doc_height
.saturating_sub(current_line)
.saturating_sub(inner.height as usize),
),
};
offset.anchor = match preview_scroll_offset.0 {
Direction::Backward => doc
.text()
.slice(..)
.line_to_char(current_line.saturating_sub(preview_scroll_offset.1)),
Direction::Forward => doc
.text()
.slice(..)
.line_to_char(current_line.saturating_add(preview_scroll_offset.1)),
};
let syntax_highlights = EditorView::doc_syntax_highlights( let syntax_highlights = EditorView::doc_syntax_highlights(
doc, doc,
offset.anchor, offset.anchor,
@ -970,6 +1047,8 @@ impl<T: 'static + Send + Sync, D: 'static + Send + Sync> Picker<T, D> {
decorations.add_decoration(draw_highlight); decorations.add_decoration(draw_highlight);
} }
current_line = doc.text().slice(..).char_to_line(offset.anchor);
render_document( render_document(
surface, surface,
inner, inner,
@ -982,6 +1061,39 @@ impl<T: 'static + Send + Sync, D: 'static + Send + Sync> Picker<T, D> {
&cx.editor.theme, &cx.editor.theme,
decorations, decorations,
); );
self.preview_scroll_offset = preview_scroll_offset;
let win_height = inner.height as usize;
let len = doc_height;
let fits = len <= win_height;
let scroll = current_line;
let scroll_style = cx.editor.theme.get("ui.menu.scroll");
const fn div_ceil(a: usize, b: usize) -> usize {
(a + b - 1) / b
}
if !fits {
let scroll_height = div_ceil(win_height.pow(2), len).min(win_height);
let scroll_line = (win_height - scroll_height) * scroll
/ std::cmp::max(1, len.saturating_sub(win_height));
let mut cell;
for i in 0..win_height {
cell = &mut surface[(inner.right() - 1, inner.top() + i as u16)];
cell.set_symbol(""); // right half block
if scroll_line <= i && i < scroll_line + scroll_height {
// Draw scroll thumb
cell.set_fg(scroll_style.fg.unwrap_or(helix_view::theme::Color::Reset));
} else {
// Draw scroll track
cell.set_fg(scroll_style.bg.unwrap_or(helix_view::theme::Color::Reset));
}
}
}
} }
} }
} }
@ -1053,10 +1165,10 @@ impl<I: 'static + Send + Sync, D: 'static + Send + Sync> Component for Picker<I,
key!(Tab) | key!(Down) | ctrl!('n') => { key!(Tab) | key!(Down) | ctrl!('n') => {
self.move_by(1, Direction::Forward); self.move_by(1, Direction::Forward);
} }
key!(PageDown) | ctrl!('d') => { key!(PageDown) | ctrl!('d') if !self.show_preview => {
self.page_down(); self.page_down();
} }
key!(PageUp) | ctrl!('u') => { key!(PageUp) | ctrl!('u') if !self.show_preview => {
self.page_up(); self.page_up();
} }
key!(Home) => { key!(Home) => {
@ -1121,6 +1233,36 @@ impl<I: 'static + Send + Sync, D: 'static + Send + Sync> Component for Picker<I,
ctrl!('t') => { ctrl!('t') => {
self.toggle_preview(); self.toggle_preview();
} }
alt!('k') | shift!(Up) if self.show_preview => {
self.move_preview_by(
ctx.editor.config().scroll_lines.unsigned_abs(),
Direction::Backward,
);
}
alt!('j') | shift!(Down) if self.show_preview => {
self.move_preview_by(
ctx.editor.config().scroll_lines.unsigned_abs(),
Direction::Forward,
);
}
alt!('u') if self.show_preview => {
self.move_preview_by(
self.preview_height.saturating_div(2) as usize,
Direction::Backward,
);
}
alt!('d') if self.show_preview => {
self.move_preview_by(
self.preview_height.saturating_div(2) as usize,
Direction::Forward,
);
}
key!(PageUp) | alt!('b') if self.show_preview => {
self.move_preview_by(self.preview_height as usize, Direction::Backward);
}
key!(PageDown) | alt!('f') if self.show_preview => {
self.move_preview_by(self.preview_height as usize, Direction::Forward);
}
_ => { _ => {
self.prompt_handle_event(event, ctx); self.prompt_handle_event(event, ctx);
} }
@ -1150,6 +1292,8 @@ impl<I: 'static + Send + Sync, D: 'static + Send + Sync> Component for Picker<I,
fn required_size(&mut self, (width, height): (u16, u16)) -> Option<(u16, u16)> { fn required_size(&mut self, (width, height): (u16, u16)) -> Option<(u16, u16)> {
self.completion_height = height.saturating_sub(4 + self.header_height()); self.completion_height = height.saturating_sub(4 + self.header_height());
self.preview_height = height.saturating_sub(2);
Some((width, height)) Some((width, height))
} }