Handle PageUp and PageDown in the completion and popups

This fixes three issues:

1) PageUp and PageDown during completion selection moved the
   viewport without cancelling the completion itself. It also
   left the last completion selected in the text.

   The change adds PageUp and PageDown handling to the existing
   movements (with the Ctrl-u and Ctrl-d aliases) to make
   the behavior of the editor and the menus consistent with
   each other.

2) Completion key actions were processed after the popup key
   actions. This prevented the PageUp / PageDown logic from
   being executed, because the popup swallowed them.

   The change reverses the logic, because the most active
   element should be handling the keys. Both code lens
   (space-k) and code completion now handle the page
   movements properly and consistently.

3) Popup Ctrl-u and Ctrl-d movements did not have the
   PageUp and PageDown aliases defined. This was confusing
   for new users as the editor itself recognizes those.

Signed-off-by: Martin Sivak <mars@montik.net>
This commit is contained in:
Martin Sivak 2025-03-08 09:58:18 +01:00
parent dc4761ad3a
commit 536033faad
2 changed files with 58 additions and 23 deletions

View file

@ -102,6 +102,14 @@ impl<T: Item> Menu<T> {
self.adjust_scroll(); self.adjust_scroll();
} }
pub fn move_half_page_up(&mut self) {
let len = self.matches.len();
let max_index = len.saturating_sub((self.size.1 as usize / 2).max(1));
let pos = self.cursor.map_or(max_index, |i| (i + max_index) % len) % len;
self.cursor = Some(pos);
self.adjust_scroll();
}
pub fn move_down(&mut self) { pub fn move_down(&mut self) {
let len = self.matches.len(); let len = self.matches.len();
let pos = self.cursor.map_or(0, |i| i + 1) % len; let pos = self.cursor.map_or(0, |i| i + 1) % len;
@ -109,6 +117,16 @@ impl<T: Item> Menu<T> {
self.adjust_scroll(); self.adjust_scroll();
} }
pub fn move_half_page_down(&mut self) {
let len = self.matches.len();
let pos = self
.cursor
.map_or(0, |i| i + (self.size.1 as usize / 2).max(1))
% len;
self.cursor = Some(pos);
self.adjust_scroll();
}
fn recalculate_size(&mut self, viewport: (u16, u16)) { fn recalculate_size(&mut self, viewport: (u16, u16)) {
let n = self let n = self
.options .options
@ -251,6 +269,18 @@ impl<T: Item + 'static> Component for Menu<T> {
(self.callback_fn)(cx.editor, self.selection(), MenuEvent::Update); (self.callback_fn)(cx.editor, self.selection(), MenuEvent::Update);
return EventResult::Consumed(None); return EventResult::Consumed(None);
} }
key!(PageUp) | ctrl!('u') => {
// page up moves back in the completion choice (including updating the doc)
self.move_half_page_up();
(self.callback_fn)(cx.editor, self.selection(), MenuEvent::Update);
return EventResult::Consumed(None);
}
key!(PageDown) | ctrl!('d') => {
// page down advances completion choice (including updating the doc)
self.move_half_page_down();
(self.callback_fn)(cx.editor, self.selection(), MenuEvent::Update);
return EventResult::Consumed(None);
}
key!(Enter) => { key!(Enter) => {
if let Some(selection) = self.selection() { if let Some(selection) = self.selection() {
(self.callback_fn)(cx.editor, Some(selection), MenuEvent::Validate); (self.callback_fn)(cx.editor, Some(selection), MenuEvent::Validate);

View file

@ -270,34 +270,39 @@ impl<T: Component> Component for Popup<T> {
compositor.remove(self.id.as_ref()); compositor.remove(self.id.as_ref());
}); });
match key { // Code completion handles arrows and page up/down itself,
// esc or ctrl-c aborts the completion and closes the menu // but code lens does not. First check whether content knows
key!(Esc) | ctrl!('c') => { // about the key event. When not, check the default keys.
let _ = self.contents.handle_event(event, cx); match self.contents.handle_event(event, cx) {
EventResult::Consumed(Some(close_fn)) EventResult::Ignored(fn_once) => {
} match key {
ctrl!('d') => { // esc or ctrl-c aborts the completion and closes the menu
self.scroll_half_page_down(); key!(Esc) | ctrl!('c') => {
EventResult::Consumed(None) let _ = self.contents.handle_event(event, cx);
} EventResult::Consumed(Some(close_fn))
ctrl!('u') => { }
self.scroll_half_page_up(); key!(PageDown) | ctrl!('d') => {
EventResult::Consumed(None) self.scroll_half_page_down();
} EventResult::Consumed(None)
_ => { }
let contents_event_result = self.contents.handle_event(event, cx); key!(PageUp) | ctrl!('u') => {
self.scroll_half_page_up();
EventResult::Consumed(None)
}
_ => {
// for some events, we want to process them but send ignore, specifically all input except
// tab/enter/ctrl-k or whatever will confirm the selection/ ctrl-n/ctrl-p for scroll.
if self.auto_close { if self.auto_close {
if let EventResult::Ignored(None) = contents_event_result { EventResult::Ignored(Some(close_fn))
return EventResult::Ignored(Some(close_fn)); } else {
EventResult::Ignored(fn_once)
}
} }
} }
contents_event_result
} }
ev => ev,
} }
// for some events, we want to process them but send ignore, specifically all input except
// tab/enter/ctrl-k or whatever will confirm the selection/ ctrl-n/ctrl-p for scroll.
} }
fn render(&mut self, viewport: Rect, surface: &mut Surface, cx: &mut Context) { fn render(&mut self, viewport: Rect, surface: &mut Surface, cx: &mut Context) {