Create surface only once for given drawing area size

This commit is contained in:
daa 2018-02-24 23:36:40 +03:00
parent bec70d298a
commit bb61b31830

View File

@ -6,6 +6,7 @@ use std::thread;
use std::time::Duration; use std::time::Duration;
use cairo; use cairo;
use cairo::prelude::*;
use pango::{FontDescription, LayoutExt}; use pango::{FontDescription, LayoutExt};
use gdk; use gdk;
use gdk::{EventButton, EventMotion, EventScroll, EventType, ModifierType}; use gdk::{EventButton, EventMotion, EventScroll, EventType, ModifierType};
@ -49,6 +50,34 @@ macro_rules! idle_cb_call {
) )
} }
/// Double buffer surface
pub struct Surface {
surface: cairo::Surface,
ctx: cairo::Context,
width: i32,
height: i32,
}
impl Surface {
pub fn new(drawing_area: &gtk::DrawingArea) -> Self {
let alloc = drawing_area.get_allocation();
let surface = drawing_area
.get_window()
.unwrap()
.create_similar_surface(cairo::Content::Color, alloc.width, alloc.height)
.unwrap();
let ctx = cairo::Context::new(&surface);
Surface {
surface,
ctx,
width: alloc.width,
height: alloc.height,
}
}
}
pub struct State { pub struct State {
pub model: UiModel, pub model: UiModel,
pub color_model: ColorModel, pub color_model: ColorModel,
@ -60,6 +89,8 @@ pub struct State {
popup_menu: RefCell<PopupMenu>, popup_menu: RefCell<PopupMenu>,
settings: Rc<RefCell<Settings>>, settings: Rc<RefCell<Settings>>,
surface: Option<Surface>,
resize_request: (i64, i64), resize_request: (i64, i64),
resize_timer: Rc<Cell<Option<glib::SourceId>>>, resize_timer: Rc<Cell<Option<glib::SourceId>>>,
@ -97,6 +128,8 @@ impl State {
popup_menu, popup_menu,
settings, settings,
surface: None,
resize_request: (-1, -1), resize_request: (-1, -1),
resize_timer: Rc::new(Cell::new(None)), resize_timer: Rc::new(Cell::new(None)),
@ -119,6 +152,18 @@ impl State {
} }
} }
fn resize_surface(&mut self) {
if let Some(Surface { width, height, .. }) = self.surface {
let alloc = self.drawing_area.get_allocation();
if width != alloc.width || height != alloc.height {
self.surface = Some(Surface::new(&self.drawing_area));
}
} else {
self.surface = Some(Surface::new(&self.drawing_area));
}
}
/// Return NeovimRef only if vim in non blocking state /// Return NeovimRef only if vim in non blocking state
/// ///
/// Note that this call also do neovim api call get_mode /// Note that this call also do neovim api call get_mode
@ -525,7 +570,11 @@ impl Shell {
let ref_state = self.state.clone(); let ref_state = self.state.clone();
state.drawing_area.connect_configure_event(move |_, ev| { state.drawing_area.connect_configure_event(move |_, ev| {
debug!("configure_event {:?}", ev.get_size()); debug!("configure_event {:?}", ev.get_size());
ref_state.borrow_mut().try_nvim_resize();
let mut state = ref_state.borrow_mut();
state.resize_surface();
state.try_nvim_resize();
false false
}); });
@ -738,17 +787,14 @@ fn gtk_motion_notify(shell: &mut State, ui_state: &mut UiState, ev: &EventMotion
fn gtk_draw(state_arc: &Arc<UiMutex<State>>, ctx: &cairo::Context) -> Inhibit { 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() {
// create buffer surface
let (x1, y1, x2, y2) = ctx.clip_extents();
let alloc = state.drawing_area.get_allocation();
let surface = state
.drawing_area
.get_window()
.unwrap()
.create_similar_surface(cairo::Content::Color, alloc.width, alloc.height)
.unwrap();
let buf_ctx = cairo::Context::new(&surface); let (x1, y1, x2, y2) = ctx.clip_extents();
let surface = state.surface.as_ref().unwrap();
let buf_ctx = &surface.ctx;
surface.surface.flush();
buf_ctx.save();
buf_ctx.rectangle(x1, y1, x2 - x1, y2 - y1); buf_ctx.rectangle(x1, y1, x2 - x1, y2 - y1);
buf_ctx.clip(); buf_ctx.clip();
@ -761,8 +807,9 @@ fn gtk_draw(state_arc: &Arc<UiMutex<State>>, ctx: &cairo::Context) -> Inhibit {
&state.mode, &state.mode,
); );
ctx.set_source_surface(&surface, 0.0, 0.0); ctx.set_source_surface(&surface.surface, 0.0, 0.0);
ctx.paint(); ctx.paint();
buf_ctx.restore();
} else if state.nvim.is_initializing() { } else if state.nvim.is_initializing() {
draw_initializing(&*state, ctx); draw_initializing(&*state, ctx);
} }