diff options
author | Blaž Hrastnik | 2021-03-02 09:24:24 +0000 |
---|---|---|
committer | Blaž Hrastnik | 2021-03-02 09:24:24 +0000 |
commit | 0b85c16be91d9326876b2e4a1ae6bdc1381d700d (patch) | |
tree | 4bbb758097d6b3aecb8791b061bd49a59dc5ad7a /helix-term/src/ui/popup.rs | |
parent | 11c4e0b05325d8fd8046a764ec4b9e655486313d (diff) |
ui: Share popup code with menu.
Menu is now just wrapped in a popup.
Diffstat (limited to 'helix-term/src/ui/popup.rs')
-rw-r--r-- | helix-term/src/ui/popup.rs | 47 |
1 files changed, 24 insertions, 23 deletions
diff --git a/helix-term/src/ui/popup.rs b/helix-term/src/ui/popup.rs index 673321dc..ba32e6b5 100644 --- a/helix-term/src/ui/popup.rs +++ b/helix-term/src/ui/popup.rs @@ -16,28 +16,28 @@ use helix_view::Editor; // a width/height hint. maybe Popup(Box<Component>) pub struct Popup { - contents: String, - position: Position, + contents: Box<dyn Component>, + position: Option<Position>, } impl Popup { // TODO: it's like a slimmed down picker, share code? (picker = menu + prompt with different // rendering) - pub fn new(contents: String) -> Self { + pub fn new(contents: Box<dyn Component>) -> Self { Self { contents, - position: Position::default(), + position: None, } } - pub fn set_position(&mut self, pos: Position) { + pub fn set_position(&mut self, pos: Option<Position>) { self.position = pos; } } impl Component for Popup { fn handle_event(&mut self, event: Event, cx: &mut Context) -> EventResult { - let event = match event { + let key = match event { Event::Key(event) => event, _ => return EventResult::Ignored, }; @@ -49,7 +49,7 @@ impl Component for Popup { }, ))); - match event { + match key { // esc or ctrl-c aborts the completion and closes the menu KeyEvent { code: KeyCode::Esc, .. @@ -60,29 +60,37 @@ impl Component for Popup { } => { return close_fn; } - _ => (), + _ => self.contents.handle_event(event, cx), } // 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. - // EventResult::Consumed(None) - EventResult::Consumed(None) } fn render(&self, viewport: Rect, surface: &mut Surface, cx: &mut Context) { use tui::text::Text; use tui::widgets::{Paragraph, Widget, Wrap}; - let contents = Text::from(self.contents.clone()); + let position = self + .position + .or_else(|| cx.editor.cursor_position()) + .unwrap_or_default(); - let width = contents.width().min(150) as u16; - let height = contents.height().min(13) as u16; + let (width, height) = self + .contents + .size_hint(viewport) + .expect("Component needs size_hint implemented in order to be embedded in a popup"); + + let width = width.min(150) as u16; + let height = height.min(13) as u16; // -- make sure frame doesn't stick out of bounds - let mut rel_x = self.position.col as u16; - let mut rel_y = self.position.row as u16; + let mut rel_x = position.col as u16; + let mut rel_y = position.row as u16; if viewport.width <= rel_x + width { rel_x -= ((rel_x + width) - viewport.width) }; + // TODO: be able to specify orientation preference. We want above for most popups, below + // for menus/autocomplete. if height <= rel_y { rel_y -= height // position above point } else { @@ -104,13 +112,6 @@ impl Component for Popup { } } - // -- Render the contents: - - let style = Style::default().fg(Color::Rgb(164, 160, 232)); // lavender - - let par = Paragraph::new(contents).wrap(Wrap { trim: false }); - // .scroll(x, y) offsets - - par.render(area, surface); + self.contents.render(area, surface, cx); } } |