One step to cmd line implementation
This commit is contained in:
		
							parent
							
								
									519c279c8d
								
							
						
					
					
						commit
						d6aa19510c
					
				
							
								
								
									
										132
									
								
								src/cmd_line.rs
									
									
									
									
									
								
							
							
						
						
									
										132
									
								
								src/cmd_line.rs
									
									
									
									
									
								
							| @ -1,11 +1,19 @@ | |||||||
| use std::collections::HashMap; | use std::collections::HashMap; | ||||||
|  | use std::rc::Rc; | ||||||
|  | use std::sync::Arc; | ||||||
|  | use std::cell::RefCell; | ||||||
| 
 | 
 | ||||||
| use gtk; | use gtk; | ||||||
| use gtk::prelude::*; | use gtk::prelude::*; | ||||||
|  | use cairo; | ||||||
| 
 | 
 | ||||||
| use neovim_lib::Value; | use neovim_lib::Value; | ||||||
| 
 | 
 | ||||||
| use ui_model::{UiModel, Attrs}; | use ui_model::{UiModel, Attrs}; | ||||||
|  | use ui::UiMutex; | ||||||
|  | use render; | ||||||
|  | use shell; | ||||||
|  | use cursor; | ||||||
| 
 | 
 | ||||||
| pub struct Level { | pub struct Level { | ||||||
|     model: UiModel, |     model: UiModel, | ||||||
| @ -21,9 +29,9 @@ impl Level { | |||||||
|         firstc: String, |         firstc: String, | ||||||
|         prompt: String, |         prompt: String, | ||||||
|         indent: u64, |         indent: u64, | ||||||
|         level: u64, |  | ||||||
|     ) -> Self { |     ) -> Self { | ||||||
|         //TODO: double width chars
 |         //TODO: double width chars
 | ||||||
|  |         //TODO: im
 | ||||||
| 
 | 
 | ||||||
|         let prompt = prompt_lines(firstc, prompt, indent); |         let prompt = prompt_lines(firstc, prompt, indent); | ||||||
|         let content: Vec<(Attrs, Vec<char>)> = content |         let content: Vec<(Attrs, Vec<char>)> = content | ||||||
| @ -31,14 +39,44 @@ impl Level { | |||||||
|             .map(|c| (Attrs::from_value_map(&c.0), c.1.chars().collect())) |             .map(|c| (Attrs::from_value_map(&c.0), c.1.chars().collect())) | ||||||
|             .collect(); |             .collect(); | ||||||
| 
 | 
 | ||||||
|         let width = (content.iter().map(|c| c.1.len()).count() + prompt.last().map_or(0, |p| p.len())) as u64; |         let width = (content.iter().map(|c| c.1.len()).count() + | ||||||
|  |                          prompt.last().map_or(0, |p| p.len())) as u64; | ||||||
|         let columns = ((width / Level::COLUMNS_STEP) + 1) * Level::COLUMNS_STEP; |         let columns = ((width / Level::COLUMNS_STEP) + 1) * Level::COLUMNS_STEP; | ||||||
|         let rows = ((prompt.len() as u64 / Level::ROWS_STEP) + 1) * Level::ROWS_STEP; |         let rows = ((prompt.len() as u64 / Level::ROWS_STEP) + 1) * Level::ROWS_STEP; | ||||||
| 
 | 
 | ||||||
|         let model = UiModel::new(rows, columns); |         let mut model = UiModel::new(rows, columns); | ||||||
|  | 
 | ||||||
|  |         for (row_idx, prompt_line) in prompt.iter().enumerate() { | ||||||
|  |             for (col_idx, &ch) in prompt_line.iter().enumerate() { | ||||||
|  |                 model.set_cursor(row_idx, col_idx); | ||||||
|  |                 model.put(ch, false, None); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         let mut col_idx = 0; | ||||||
|  |         let row_idx = if prompt.len() > 0 { | ||||||
|  |             prompt.len() - 1 | ||||||
|  |         } else { | ||||||
|  |             0 | ||||||
|  |         }; | ||||||
|  |         for (attr, ch_list) in content { | ||||||
|  |             for ch in ch_list { | ||||||
|  |                 model.set_cursor(row_idx, col_idx); | ||||||
|  |                 model.put(ch, false, Some(&attr)); | ||||||
|  |                 col_idx += 1; | ||||||
|  |             } | ||||||
|  |         } | ||||||
| 
 | 
 | ||||||
|         Level { model } |         Level { model } | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     fn update_cache(&mut self, render_state: &shell::RenderState) { | ||||||
|  |         render::shape_dirty( | ||||||
|  |             &render_state.font_ctx, | ||||||
|  |             &mut self.model, | ||||||
|  |             &render_state.color_model, | ||||||
|  |         ); | ||||||
|  |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| fn prompt_lines(firstc: String, prompt: String, indent: u64) -> Vec<Vec<char>> { | fn prompt_lines(firstc: String, prompt: String, indent: u64) -> Vec<Vec<char>> { | ||||||
| @ -51,31 +89,109 @@ fn prompt_lines(firstc: String, prompt: String, indent: u64) -> Vec<Vec<char>> { | |||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | struct State { | ||||||
|  |     levels: Vec<Level>, | ||||||
|  |     render_state: Rc<RefCell<shell::RenderState>>, | ||||||
|  |     drawing_area: gtk::DrawingArea, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl State { | ||||||
|  |     fn new(drawing_area: gtk::DrawingArea, render_state: Rc<RefCell<shell::RenderState>>) -> Self { | ||||||
|  |         State { | ||||||
|  |             levels: Vec::new(), | ||||||
|  |             render_state, | ||||||
|  |             drawing_area, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl cursor::CursorRedrawCb for State { | ||||||
|  |     fn queue_redraw_cursor(&mut self) { | ||||||
|  |         // TODO: implement
 | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
| pub struct CmdLine { | pub struct CmdLine { | ||||||
|     popover: gtk::Popover, |     popover: gtk::Popover, | ||||||
|     levels: Vec<Level>, |     displyed: bool, | ||||||
|  |     state: Arc<UiMutex<State>>, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl CmdLine { | impl CmdLine { | ||||||
|     pub fn new(drawing: >k::DrawingArea) -> Self { |     pub fn new(drawing: >k::DrawingArea, render_state: Rc<RefCell<shell::RenderState>>) -> Self { | ||||||
|         let popover = gtk::Popover::new(Some(drawing)); |         let popover = gtk::Popover::new(Some(drawing)); | ||||||
|         popover.set_modal(false); |         popover.set_modal(false); | ||||||
|         let edit_frame = gtk::Frame::new(None); |         let edit_frame = gtk::Frame::new(None); | ||||||
|         edit_frame.set_shadow_type(gtk::ShadowType::In); |         edit_frame.set_shadow_type(gtk::ShadowType::In); | ||||||
|         let drawing_area = gtk::DrawingArea::new(); |         let drawing_area = gtk::DrawingArea::new(); | ||||||
|  |         drawing_area.set_size_request(50, 50); | ||||||
|         edit_frame.add(&drawing_area); |         edit_frame.add(&drawing_area); | ||||||
|         edit_frame.show_all(); |         edit_frame.show_all(); | ||||||
| 
 | 
 | ||||||
|         popover.add(&edit_frame); |         popover.add(&edit_frame); | ||||||
| 
 | 
 | ||||||
|  |         let state = Arc::new(UiMutex::new(State::new(drawing_area.clone(), render_state))); | ||||||
|  |         let weak_cb = Arc::downgrade(&state); | ||||||
|  |         let cursor = cursor::Cursor::new(weak_cb); | ||||||
|  | 
 | ||||||
|  |         drawing_area.connect_draw( | ||||||
|  |             clone!(state => move |_, ctx| gtk_draw(ctx, &state, &cursor)), | ||||||
|  |         ); | ||||||
|  | 
 | ||||||
|         CmdLine { |         CmdLine { | ||||||
|             levels: Vec::new(), |  | ||||||
|             popover, |             popover, | ||||||
|  |             state, | ||||||
|  |             displyed: false, | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     pub fn show_level(&mut self, level: Level) { |     pub fn show_level( | ||||||
|         self.levels.push(level); |         &mut self, | ||||||
|  |         content: Vec<(HashMap<String, Value>, String)>, | ||||||
|  |         pos: u64, | ||||||
|  |         firstc: String, | ||||||
|  |         prompt: String, | ||||||
|  |         indent: u64, | ||||||
|  |         level_idx: u64, | ||||||
|  |     ) { | ||||||
|  |         let mut state = self.state.borrow_mut(); | ||||||
|  | 
 | ||||||
|  |         let mut level = Level::from(content, pos, firstc, prompt, indent); | ||||||
|  |         level.update_cache(&*state.render_state.borrow()); | ||||||
|  | 
 | ||||||
|  |         if level_idx as usize == state.levels.len() { | ||||||
|  |             // TODO: update level
 | ||||||
|  |             state.levels.pop(); | ||||||
|  |         } | ||||||
|  |         state.levels.push(level); | ||||||
|  |         if !self.displyed { | ||||||
|  |             self.displyed = true; | ||||||
|             self.popover.popup(); |             self.popover.popup(); | ||||||
|  |         } else { | ||||||
|  |             state.drawing_area.queue_draw() | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | fn gtk_draw( | ||||||
|  |     ctx: &cairo::Context, | ||||||
|  |     state: &Arc<UiMutex<State>>, | ||||||
|  |     cursor: &cursor::Cursor<State>, | ||||||
|  | ) -> Inhibit { | ||||||
|  |     let state = state.borrow(); | ||||||
|  |     let level = state.levels.last(); | ||||||
|  | 
 | ||||||
|  |     if let Some(level) = level { | ||||||
|  |         let render_state = state.render_state.borrow(); | ||||||
|  | 
 | ||||||
|  |         render::render( | ||||||
|  |             ctx, | ||||||
|  |             cursor, | ||||||
|  |             &render_state.font_ctx, | ||||||
|  |             &level.model, | ||||||
|  |             &render_state.color_model, | ||||||
|  |             &render_state.mode, | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |     Inhibit(false) | ||||||
|  | } | ||||||
|  | |||||||
| @ -1,10 +1,8 @@ | |||||||
| use cairo; | use cairo; | ||||||
| use color::Color; | use color::Color; | ||||||
| use ui::UiMutex; | use ui::UiMutex; | ||||||
| use shell; |  | ||||||
| use mode; | use mode; | ||||||
| use nvim; | use nvim; | ||||||
| use nvim::{RepaintMode, RedrawEvents}; |  | ||||||
| use std::sync::{Arc, Weak}; | use std::sync::{Arc, Weak}; | ||||||
| use render; | use render; | ||||||
| use render::CellMetrics; | use render::CellMetrics; | ||||||
| @ -44,20 +42,20 @@ enum AnimPhase { | |||||||
|     Busy, |     Busy, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| struct State { | struct State<CB: CursorRedrawCb> { | ||||||
|     alpha: Alpha, |     alpha: Alpha, | ||||||
|     anim_phase: AnimPhase, |     anim_phase: AnimPhase, | ||||||
|     shell: Weak<UiMutex<shell::State>>, |     redraw_cb: Weak<UiMutex<CB>>, | ||||||
| 
 | 
 | ||||||
|     timer: Option<glib::SourceId>, |     timer: Option<glib::SourceId>, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl State { | impl <CB: CursorRedrawCb> State <CB> { | ||||||
|     fn new(shell: Weak<UiMutex<shell::State>>) -> State { |     fn new(redraw_cb: Weak<UiMutex<CB>>) -> Self { | ||||||
|         State { |         State { | ||||||
|             alpha: Alpha(1.0), |             alpha: Alpha(1.0), | ||||||
|             anim_phase: AnimPhase::Shown, |             anim_phase: AnimPhase::Shown, | ||||||
|             shell: shell, |             redraw_cb, | ||||||
|             timer: None, |             timer: None, | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @ -72,13 +70,13 @@ impl State { | |||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| pub struct Cursor { | pub struct Cursor <CB: CursorRedrawCb> { | ||||||
|     state: Arc<UiMutex<State>>, |     state: Arc<UiMutex<State<CB>>>, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl Cursor { | impl <CB: CursorRedrawCb + 'static> Cursor <CB> { | ||||||
|     pub fn new(shell: Weak<UiMutex<shell::State>>) -> Cursor { |     pub fn new(redraw_cb: Weak<UiMutex<CB>>) -> Self { | ||||||
|         Cursor { state: Arc::new(UiMutex::new(State::new(shell))) } |         Cursor { state: Arc::new(UiMutex::new(State::new(redraw_cb))) } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     pub fn start(&mut self) { |     pub fn start(&mut self) { | ||||||
| @ -205,7 +203,8 @@ fn cursor_rect( | |||||||
|         (line_y, cursor_width, line_height) |         (line_y, cursor_width, line_height) | ||||||
|     } |     } | ||||||
| } | } | ||||||
| fn anim_step(state: &Arc<UiMutex<State>>) -> glib::Continue { | 
 | ||||||
|  | fn anim_step<CB: CursorRedrawCb + 'static> (state: &Arc<UiMutex<State<CB>>>) -> glib::Continue { | ||||||
|     let mut mut_state = state.borrow_mut(); |     let mut mut_state = state.borrow_mut(); | ||||||
| 
 | 
 | ||||||
|     let next_event = match mut_state.anim_phase { |     let next_event = match mut_state.anim_phase { | ||||||
| @ -240,10 +239,9 @@ fn anim_step(state: &Arc<UiMutex<State>>) -> glib::Continue { | |||||||
|         AnimPhase::Busy => None, |         AnimPhase::Busy => None, | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     let shell = mut_state.shell.upgrade().unwrap(); |     let redraw_cb = mut_state.redraw_cb.upgrade().unwrap(); | ||||||
|     let mut shell = shell.borrow_mut(); |     let mut redraw_cb = redraw_cb.borrow_mut(); | ||||||
|     let point = shell.model.cur_point(); |     redraw_cb.queue_redraw_cursor(); | ||||||
|     shell.on_redraw(&RepaintMode::Area(point)); |  | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|     if let Some(timeout) = next_event { |     if let Some(timeout) = next_event { | ||||||
| @ -257,7 +255,7 @@ fn anim_step(state: &Arc<UiMutex<State>>) -> glib::Continue { | |||||||
| 
 | 
 | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl Drop for Cursor { | impl <CB: CursorRedrawCb> Drop for Cursor<CB> { | ||||||
|     fn drop(&mut self) { |     fn drop(&mut self) { | ||||||
|         if let Some(timer_id) = self.state.borrow().timer { |         if let Some(timer_id) = self.state.borrow().timer { | ||||||
|             glib::source_remove(timer_id); |             glib::source_remove(timer_id); | ||||||
| @ -265,6 +263,10 @@ impl Drop for Cursor { | |||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | pub trait CursorRedrawCb { | ||||||
|  |     fn queue_redraw_cursor(&mut self); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| #[cfg(test)] | #[cfg(test)] | ||||||
| mod tests { | mod tests { | ||||||
|     use super::*; |     use super::*; | ||||||
|  | |||||||
| @ -15,9 +15,9 @@ use cursor; | |||||||
| use pangocairo::CairoContextExt; | use pangocairo::CairoContextExt; | ||||||
| use ui_model; | use ui_model; | ||||||
| 
 | 
 | ||||||
| pub fn render( | pub fn render<RC: cursor::CursorRedrawCb + 'static>( | ||||||
|     ctx: &cairo::Context, |     ctx: &cairo::Context, | ||||||
|     cursor: &cursor::Cursor, |     cursor: &cursor::Cursor<RC>, | ||||||
|     font_ctx: &context::Context, |     font_ctx: &context::Context, | ||||||
|     ui_model: &ui_model::UiModel, |     ui_model: &ui_model::UiModel, | ||||||
|     color_model: &color::ColorModel, |     color_model: &color::ColorModel, | ||||||
|  | |||||||
							
								
								
									
										140
									
								
								src/shell.rs
									
									
									
									
									
								
							
							
						
						
									
										140
									
								
								src/shell.rs
									
									
									
									
									
								
							| @ -26,7 +26,7 @@ use nvim::{self, RedrawEvents, GuiApi, RepaintMode, ErrorReport, NeovimClient, N | |||||||
|            NeovimClientAsync}; |            NeovimClientAsync}; | ||||||
| 
 | 
 | ||||||
| use input::{self, keyval_to_input_string}; | use input::{self, keyval_to_input_string}; | ||||||
| use cursor::Cursor; | use cursor::{Cursor, CursorRedrawCb}; | ||||||
| use ui::UiMutex; | use ui::UiMutex; | ||||||
| use popup_menu::{self, PopupMenu}; | use popup_menu::{self, PopupMenu}; | ||||||
| use tabline::Tabline; | use tabline::Tabline; | ||||||
| @ -58,19 +58,32 @@ enum ResizeState { | |||||||
|     Wait, |     Wait, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | pub struct RenderState { | ||||||
|  |     pub font_ctx: render::Context, | ||||||
|  |     pub color_model: ColorModel, | ||||||
|  |     pub mode: mode::Mode, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl RenderState { | ||||||
|  |     pub fn new() -> Self { | ||||||
|  |         RenderState { | ||||||
|  |             font_ctx: render::Context::new(FontDescription::from_string(DEFAULT_FONT_NAME)), | ||||||
|  |             color_model: ColorModel::new(), | ||||||
|  |             mode: mode::Mode::new(), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
| pub struct State { | pub struct State { | ||||||
|     pub model: UiModel, |     pub model: UiModel, | ||||||
|     pub color_model: ColorModel, |  | ||||||
|     cur_attrs: Option<Attrs>, |     cur_attrs: Option<Attrs>, | ||||||
|     mouse_enabled: bool, |     mouse_enabled: bool, | ||||||
|     nvim: Rc<NeovimClient>, |     nvim: Rc<NeovimClient>, | ||||||
|     pub font_ctx: render::Context, |     cursor: Option<Cursor<State>>, | ||||||
|     cursor: Option<Cursor>, |  | ||||||
|     popup_menu: PopupMenu, |     popup_menu: PopupMenu, | ||||||
|     cmd_line: CmdLine, |     cmd_line: CmdLine, | ||||||
|     settings: Rc<RefCell<Settings>>, |     settings: Rc<RefCell<Settings>>, | ||||||
| 
 |     render_state: Rc<RefCell<RenderState>>, | ||||||
|     pub mode: mode::Mode, |  | ||||||
| 
 | 
 | ||||||
|     stack: gtk::Stack, |     stack: gtk::Stack, | ||||||
|     drawing_area: gtk::DrawingArea, |     drawing_area: gtk::DrawingArea, | ||||||
| @ -89,23 +102,20 @@ pub struct State { | |||||||
| impl State { | impl State { | ||||||
|     pub fn new(settings: Rc<RefCell<Settings>>, options: ShellOptions) -> State { |     pub fn new(settings: Rc<RefCell<Settings>>, options: ShellOptions) -> State { | ||||||
|         let drawing_area = gtk::DrawingArea::new(); |         let drawing_area = gtk::DrawingArea::new(); | ||||||
|  |         let render_state = Rc::new(RefCell::new(RenderState::new())); | ||||||
|         let popup_menu = PopupMenu::new(&drawing_area); |         let popup_menu = PopupMenu::new(&drawing_area); | ||||||
|         let cmd_line = CmdLine::new(&drawing_area); |         let cmd_line = CmdLine::new(&drawing_area, render_state.clone()); | ||||||
|         let font_ctx = render::Context::new(FontDescription::from_string(DEFAULT_FONT_NAME)); |  | ||||||
| 
 | 
 | ||||||
|         State { |         State { | ||||||
|             model: UiModel::empty(), |             model: UiModel::empty(), | ||||||
|             color_model: ColorModel::new(), |  | ||||||
|             nvim: Rc::new(NeovimClient::new()), |             nvim: Rc::new(NeovimClient::new()), | ||||||
|             cur_attrs: None, |             cur_attrs: None, | ||||||
|             mouse_enabled: true, |             mouse_enabled: true, | ||||||
|             font_ctx, |  | ||||||
|             cursor: None, |             cursor: None, | ||||||
|             popup_menu, |             popup_menu, | ||||||
|             cmd_line, |             cmd_line, | ||||||
|             settings, |             settings, | ||||||
| 
 |             render_state, | ||||||
|             mode: mode::Mode::new(), |  | ||||||
| 
 | 
 | ||||||
|             // UI
 |             // UI
 | ||||||
|             stack: gtk::Stack::new(), |             stack: gtk::Stack::new(), | ||||||
| @ -162,12 +172,10 @@ impl State { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     pub fn get_font_desc(&self) -> &FontDescription { |  | ||||||
|         self.font_ctx.font_description() |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     pub fn set_font_desc(&mut self, desc: &str) { |     pub fn set_font_desc(&mut self, desc: &str) { | ||||||
|         self.font_ctx.update(FontDescription::from_string(desc)); |         self.render_state.borrow_mut().font_ctx.update( | ||||||
|  |             FontDescription::from_string(desc), | ||||||
|  |         ); | ||||||
|         self.model.clear_glyphs(); |         self.model.clear_glyphs(); | ||||||
|         self.try_nvim_resize(); |         self.try_nvim_resize(); | ||||||
|         self.on_redraw(&RepaintMode::All); |         self.on_redraw(&RepaintMode::All); | ||||||
| @ -207,18 +215,25 @@ impl State { | |||||||
| 
 | 
 | ||||||
|         self.update_dirty_glyphs(); |         self.update_dirty_glyphs(); | ||||||
| 
 | 
 | ||||||
|  |         let render_state = self.render_state.borrow(); | ||||||
|  |         let cell_metrics = render_state.font_ctx.cell_metrics(); | ||||||
|  | 
 | ||||||
|         for mut rect in rects { |         for mut rect in rects { | ||||||
|             rect.extend_by_items(&self.model); |             rect.extend_by_items(&self.model); | ||||||
| 
 | 
 | ||||||
|             let (x, y, width, height) = |             let (x, y, width, height) = rect.to_area_extend_ink(&self.model, cell_metrics); | ||||||
|                 rect.to_area_extend_ink(&self.model, self.font_ctx.cell_metrics()); |  | ||||||
|             self.drawing_area.queue_draw_area(x, y, width, height); |             self.drawing_area.queue_draw_area(x, y, width, height); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     #[inline] |     #[inline] | ||||||
|     fn update_dirty_glyphs(&mut self) { |     fn update_dirty_glyphs(&mut self) { | ||||||
|         render::shape_dirty(&self.font_ctx, &mut self.model, &self.color_model); |         let render_state = self.render_state.borrow(); | ||||||
|  |         render::shape_dirty( | ||||||
|  |             &render_state.font_ctx, | ||||||
|  |             &mut self.model, | ||||||
|  |             &render_state.color_model, | ||||||
|  |         ); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn im_commit(&self, ch: &str) { |     fn im_commit(&self, ch: &str) { | ||||||
| @ -232,7 +247,7 @@ impl State { | |||||||
|             line_height, |             line_height, | ||||||
|             char_width, |             char_width, | ||||||
|             .. |             .. | ||||||
|         } = self.font_ctx.cell_metrics(); |         } = self.render_state.borrow().font_ctx.cell_metrics(); | ||||||
|         let alloc = self.drawing_area.get_allocation(); |         let alloc = self.drawing_area.get_allocation(); | ||||||
|         ( |         ( | ||||||
|             (alloc.width as f64 / char_width).trunc() as usize, |             (alloc.width as f64 / char_width).trunc() as usize, | ||||||
| @ -252,7 +267,7 @@ impl State { | |||||||
|         let (row, col) = self.model.get_cursor(); |         let (row, col) = self.model.get_cursor(); | ||||||
| 
 | 
 | ||||||
|         let (x, y, width, height) = |         let (x, y, width, height) = | ||||||
|             ModelRect::point(col, row).to_area(self.font_ctx.cell_metrics()); |             ModelRect::point(col, row).to_area(self.render_state.borrow().font_ctx.cell_metrics()); | ||||||
| 
 | 
 | ||||||
|         self.im_context.set_cursor_location(&gdk::Rectangle { |         self.im_context.set_cursor_location(&gdk::Rectangle { | ||||||
|             x, |             x, | ||||||
| @ -322,7 +337,7 @@ impl State { | |||||||
|             line_height, |             line_height, | ||||||
|             char_width, |             char_width, | ||||||
|             .. |             .. | ||||||
|         } = self.font_ctx.cell_metrics(); |         } = self.render_state.borrow().font_ctx.cell_metrics(); | ||||||
| 
 | 
 | ||||||
|         let width = self.drawing_area.get_allocated_width(); |         let width = self.drawing_area.get_allocated_width(); | ||||||
|         let height = self.drawing_area.get_allocated_height(); |         let height = self.drawing_area.get_allocated_height(); | ||||||
| @ -341,7 +356,10 @@ impl State { | |||||||
|     fn edit_paste(&self, clipboard: &str) { |     fn edit_paste(&self, clipboard: &str) { | ||||||
|         let nvim = self.nvim(); |         let nvim = self.nvim(); | ||||||
|         if let Some(mut nvim) = nvim { |         if let Some(mut nvim) = nvim { | ||||||
|             if self.mode.is(&mode::NvimMode::Insert) || self.mode.is(&mode::NvimMode::Normal) { |             let render_state = self.render_state.borrow(); | ||||||
|  |             if render_state.mode.is(&mode::NvimMode::Insert) || | ||||||
|  |                 render_state.mode.is(&mode::NvimMode::Normal) | ||||||
|  |             { | ||||||
|                 let paste_code = format!("normal! \"{}P", clipboard); |                 let paste_code = format!("normal! \"{}P", clipboard); | ||||||
|                 nvim.command(&paste_code).report_err(&mut *nvim); |                 nvim.command(&paste_code).report_err(&mut *nvim); | ||||||
|             } else { |             } else { | ||||||
| @ -682,7 +700,7 @@ fn mouse_input(shell: &mut State, input: &str, state: ModifierType, position: (f | |||||||
|         line_height, |         line_height, | ||||||
|         char_width, |         char_width, | ||||||
|         .. |         .. | ||||||
|     } = shell.font_ctx.cell_metrics(); |     } = shell.render_state.borrow().font_ctx.cell_metrics(); | ||||||
|     let (x, y) = position; |     let (x, y) = position; | ||||||
|     let col = (x / char_width).trunc() as u64; |     let col = (x / char_width).trunc() as u64; | ||||||
|     let row = (y / line_height).trunc() as u64; |     let row = (y / line_height).trunc() as u64; | ||||||
| @ -722,13 +740,14 @@ fn gtk_draw(state_arc: &Arc<UiMutex<State>>, ctx: &cairo::Context) -> Inhibit { | |||||||
| 
 | 
 | ||||||
|     let state = state_arc.borrow(); |     let state = state_arc.borrow(); | ||||||
|     if state.nvim.is_initialized() { |     if state.nvim.is_initialized() { | ||||||
|  |         let render_state = state.render_state.borrow(); | ||||||
|         render::render( |         render::render( | ||||||
|             ctx, |             ctx, | ||||||
|             state.cursor.as_ref().unwrap(), |             state.cursor.as_ref().unwrap(), | ||||||
|             &state.font_ctx, |             &render_state.font_ctx, | ||||||
|             &state.model, |             &state.model, | ||||||
|             &state.color_model, |             &render_state.color_model, | ||||||
|             &state.mode, |             &render_state.mode, | ||||||
|         ); |         ); | ||||||
|     } else if state.nvim.is_initializing() { |     } else if state.nvim.is_initializing() { | ||||||
|         draw_initializing(&*state, ctx); |         draw_initializing(&*state, ctx); | ||||||
| @ -852,14 +871,16 @@ fn set_nvim_initialized(state_arc: Arc<UiMutex<State>>) { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| fn draw_initializing(state: &State, ctx: &cairo::Context) { | fn draw_initializing(state: &State, ctx: &cairo::Context) { | ||||||
|  |     let render_state = state.render_state.borrow(); | ||||||
|  |     let color_model = &render_state.color_model; | ||||||
|     let layout = ctx.create_pango_layout(); |     let layout = ctx.create_pango_layout(); | ||||||
|     let desc = state.get_font_desc(); |     let desc = render_state.font_ctx.font_description(); | ||||||
|     let alloc = state.drawing_area.get_allocation(); |     let alloc = state.drawing_area.get_allocation(); | ||||||
| 
 | 
 | ||||||
|     ctx.set_source_rgb( |     ctx.set_source_rgb( | ||||||
|         state.color_model.bg_color.0, |         color_model.bg_color.0, | ||||||
|         state.color_model.bg_color.1, |         color_model.bg_color.1, | ||||||
|         state.color_model.bg_color.2, |         color_model.bg_color.2, | ||||||
|     ); |     ); | ||||||
|     ctx.paint(); |     ctx.paint(); | ||||||
| 
 | 
 | ||||||
| @ -872,9 +893,9 @@ fn draw_initializing(state: &State, ctx: &cairo::Context) { | |||||||
| 
 | 
 | ||||||
|     ctx.move_to(x, y); |     ctx.move_to(x, y); | ||||||
|     ctx.set_source_rgb( |     ctx.set_source_rgb( | ||||||
|         state.color_model.fg_color.0, |         color_model.fg_color.0, | ||||||
|         state.color_model.fg_color.1, |         color_model.fg_color.1, | ||||||
|         state.color_model.fg_color.2, |         color_model.fg_color.2, | ||||||
|     ); |     ); | ||||||
|     ctx.update_pango_layout(&layout); |     ctx.update_pango_layout(&layout); | ||||||
|     ctx.show_pango_layout(&layout); |     ctx.show_pango_layout(&layout); | ||||||
| @ -883,11 +904,11 @@ fn draw_initializing(state: &State, ctx: &cairo::Context) { | |||||||
|     ctx.move_to(x + width as f64, y); |     ctx.move_to(x + width as f64, y); | ||||||
|     state.cursor.as_ref().unwrap().draw( |     state.cursor.as_ref().unwrap().draw( | ||||||
|         ctx, |         ctx, | ||||||
|         &state.font_ctx, |         &render_state.font_ctx, | ||||||
|         &state.mode, |         &render_state.mode, | ||||||
|         y, |         y, | ||||||
|         false, |         false, | ||||||
|         &state.color_model.bg_color, |         &color_model.bg_color, | ||||||
|     ); |     ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -947,7 +968,8 @@ impl RedrawEvents for State { | |||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         if let Some(mut nvim) = self.nvim.nvim() { |         if let Some(mut nvim) = self.nvim.nvim() { | ||||||
|             self.color_model.theme.update(&mut *nvim); |             let mut render_state = self.render_state.borrow_mut(); | ||||||
|  |             render_state.color_model.theme.update(&mut *nvim); | ||||||
|         } |         } | ||||||
|         RepaintMode::Nothing |         RepaintMode::Nothing | ||||||
|     } |     } | ||||||
| @ -981,34 +1003,38 @@ impl RedrawEvents for State { | |||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn on_update_bg(&mut self, bg: i64) -> RepaintMode { |     fn on_update_bg(&mut self, bg: i64) -> RepaintMode { | ||||||
|  |         let mut render_state = self.render_state.borrow_mut(); | ||||||
|         if bg >= 0 { |         if bg >= 0 { | ||||||
|             self.color_model.bg_color = Color::from_indexed_color(bg as u64); |             render_state.color_model.bg_color = Color::from_indexed_color(bg as u64); | ||||||
|         } else { |         } else { | ||||||
|             self.color_model.bg_color = COLOR_BLACK; |             render_state.color_model.bg_color = COLOR_BLACK; | ||||||
|         } |         } | ||||||
|         RepaintMode::Nothing |         RepaintMode::Nothing | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn on_update_fg(&mut self, fg: i64) -> RepaintMode { |     fn on_update_fg(&mut self, fg: i64) -> RepaintMode { | ||||||
|  |         let mut render_state = self.render_state.borrow_mut(); | ||||||
|         if fg >= 0 { |         if fg >= 0 { | ||||||
|             self.color_model.fg_color = Color::from_indexed_color(fg as u64); |             render_state.color_model.fg_color = Color::from_indexed_color(fg as u64); | ||||||
|         } else { |         } else { | ||||||
|             self.color_model.fg_color = COLOR_WHITE; |             render_state.color_model.fg_color = COLOR_WHITE; | ||||||
|         } |         } | ||||||
|         RepaintMode::Nothing |         RepaintMode::Nothing | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn on_update_sp(&mut self, sp: i64) -> RepaintMode { |     fn on_update_sp(&mut self, sp: i64) -> RepaintMode { | ||||||
|  |         let mut render_state = self.render_state.borrow_mut(); | ||||||
|         if sp >= 0 { |         if sp >= 0 { | ||||||
|             self.color_model.sp_color = Color::from_indexed_color(sp as u64); |             render_state.color_model.sp_color = Color::from_indexed_color(sp as u64); | ||||||
|         } else { |         } else { | ||||||
|             self.color_model.sp_color = COLOR_RED; |             render_state.color_model.sp_color = COLOR_RED; | ||||||
|         } |         } | ||||||
|         RepaintMode::Nothing |         RepaintMode::Nothing | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn on_mode_change(&mut self, mode: String, idx: u64) -> RepaintMode { |     fn on_mode_change(&mut self, mode: String, idx: u64) -> RepaintMode { | ||||||
|         self.mode.update(&mode, idx as usize); |         let mut render_state = self.render_state.borrow_mut(); | ||||||
|  |         render_state.mode.update(&mode, idx as usize); | ||||||
|         RepaintMode::Area(self.model.cur_point()) |         RepaintMode::Area(self.model.cur_point()) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @ -1034,18 +1060,19 @@ impl RedrawEvents for State { | |||||||
|         col: u64, |         col: u64, | ||||||
|     ) -> RepaintMode { |     ) -> RepaintMode { | ||||||
|         let point = ModelRect::point(col as usize, row as usize); |         let point = ModelRect::point(col as usize, row as usize); | ||||||
|         let (x, y, width, height) = point.to_area(self.font_ctx.cell_metrics()); |         let render_state = self.render_state.borrow(); | ||||||
|  |         let (x, y, width, height) = point.to_area(render_state.font_ctx.cell_metrics()); | ||||||
| 
 | 
 | ||||||
|         let context = popup_menu::PopupMenuContext { |         let context = popup_menu::PopupMenuContext { | ||||||
|             nvim: &self.nvim, |             nvim: &self.nvim, | ||||||
|             color_model: &self.color_model, |             color_model: &render_state.color_model, | ||||||
|             font_ctx: &self.font_ctx, |             font_ctx: &render_state.font_ctx, | ||||||
|             menu_items: &menu, |             menu_items: &menu, | ||||||
|             selected, |             selected, | ||||||
|             x, |             x, | ||||||
|             y, |             y, | ||||||
|             width, |             width, | ||||||
|             height |             height, | ||||||
|         }; |         }; | ||||||
| 
 | 
 | ||||||
|         self.popup_menu.show(context); |         self.popup_menu.show(context); | ||||||
| @ -1079,7 +1106,8 @@ impl RedrawEvents for State { | |||||||
|         cursor_style_enabled: bool, |         cursor_style_enabled: bool, | ||||||
|         mode_info: Vec<nvim::ModeInfo>, |         mode_info: Vec<nvim::ModeInfo>, | ||||||
|     ) -> RepaintMode { |     ) -> RepaintMode { | ||||||
|         self.mode.set_info(cursor_style_enabled, mode_info); |         let mut render_state = self.render_state.borrow_mut(); | ||||||
|  |         render_state.mode.set_info(cursor_style_enabled, mode_info); | ||||||
|         RepaintMode::Nothing |         RepaintMode::Nothing | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @ -1092,12 +1120,18 @@ impl RedrawEvents for State { | |||||||
|         indent: u64, |         indent: u64, | ||||||
|         level: u64, |         level: u64, | ||||||
|     ) -> RepaintMode { |     ) -> RepaintMode { | ||||||
|         let level = cmd_line::Level::from(content, pos, firstc, prompt, indent, level); |         self.cmd_line.show_level(content, pos, firstc, prompt, indent, level); | ||||||
|         self.cmd_line.show_level(level); |  | ||||||
|         RepaintMode::Nothing |         RepaintMode::Nothing | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | impl CursorRedrawCb for State { | ||||||
|  |     fn queue_redraw_cursor(&mut self) { | ||||||
|  |         let cur_point = self.model.cur_point(); | ||||||
|  |         self.on_redraw(&RepaintMode::Area(cur_point)); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
| impl GuiApi for State { | impl GuiApi for State { | ||||||
|     fn set_font(&mut self, font_desc: &str) { |     fn set_font(&mut self, font_desc: &str) { | ||||||
|         { |         { | ||||||
|  | |||||||
| @ -293,7 +293,7 @@ mod tests { | |||||||
| 
 | 
 | ||||||
|         model.set_cursor(1, 1); |         model.set_cursor(1, 1); | ||||||
| 
 | 
 | ||||||
|         let rect = model.put(" ", None); |         let rect = model.put(' ', false, None); | ||||||
| 
 | 
 | ||||||
|         assert_eq!(1, rect.top); |         assert_eq!(1, rect.top); | ||||||
|         assert_eq!(1, rect.left); |         assert_eq!(1, rect.left); | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 daa84
						daa84