summaryrefslogblamecommitdiff
path: root/helix-term/src/ui/info.rs
blob: 55b0e65debe6040a0e5b683c133e893c655a637c (plain) (tree)
1
2
3
4
5
6
7
8
                                            
                                         
                                   
                                                      
                         
                                                                                   




                                                                             


                                                                             
                                                                       
                                                   
                                                                            

                   
                                              
 
                                    
                                       
                                  
                                       




                                                     
                                    
 

                                          
     
use crate::compositor::{Component, Context};
use helix_view::graphics::{Margin, Rect};
use helix_view::info::Info;
use tui::buffer::Buffer as Surface;
use tui::widgets::{Block, Borders, Paragraph, Widget};

impl Component for Info {
    fn render(&mut self, viewport: Rect, surface: &mut Surface, cx: &mut Context) {
        let get_theme = |style, fallback| {
            let theme = &cx.editor.theme;
            theme.try_get(style).unwrap_or_else(|| theme.get(fallback))
        };
        let text_style = get_theme("ui.info.text", "ui.text");
        let popup_style = text_style.patch(get_theme("ui.info", "ui.popup"));

        // Calculate the area of the terminal to modify. Because we want to
        // render at the bottom right, we use the viewport's width and height
        // which evaluate to the most bottom right coordinate.
        let width = self.width + 2 + 2; // +2 for border, +2 for margin
        let height = self.height + 2; // +2 for border
        let area = viewport.intersection(Rect::new(
            viewport.width.saturating_sub(width),
            viewport.height.saturating_sub(height + 2), // +2 for statusline
            width,
            height,
        ));
        surface.clear_with(area, popup_style);

        let block = Block::default()
            .title(self.title.as_str())
            .borders(Borders::ALL)
            .border_style(popup_style);

        let margin = Margin {
            vertical: 0,
            horizontal: 1,
        };
        let inner = block.inner(area).inner(&margin);
        block.render(area, surface);

        Paragraph::new(self.text.as_str())
            .style(text_style)
            .render(inner, surface);
    }
}