Allow mouse selection

This commit is contained in:
daa 2018-04-05 23:25:18 +03:00
parent 8e8d8ba0da
commit 1306234314
3 changed files with 44 additions and 21 deletions

View File

@ -11,6 +11,7 @@ use pango;
use neovim_lib::Value; use neovim_lib::Value;
use nvim::{self, NeovimClient};
use mode; use mode;
use ui_model::{Attrs, ModelLayout}; use ui_model::{Attrs, ModelLayout};
use ui::UiMutex; use ui::UiMutex;
@ -173,6 +174,7 @@ fn prompt_lines(
} }
struct State { struct State {
nvim: Option<Rc<nvim::NeovimClient>>,
levels: Vec<Level>, levels: Vec<Level>,
block: Option<Level>, block: Option<Level>,
render_state: Rc<RefCell<shell::RenderState>>, render_state: Rc<RefCell<shell::RenderState>>,
@ -183,6 +185,7 @@ struct State {
impl State { impl State {
fn new(drawing_area: gtk::DrawingArea, render_state: Rc<RefCell<shell::RenderState>>) -> Self { fn new(drawing_area: gtk::DrawingArea, render_state: Rc<RefCell<shell::RenderState>>) -> Self {
State { State {
nvim: None,
levels: Vec::new(), levels: Vec::new(),
block: None, block: None,
render_state, render_state,
@ -292,7 +295,7 @@ impl CmdLine {
drawing_area.connect_draw(clone!(state => move |_, ctx| gtk_draw(ctx, &state))); drawing_area.connect_draw(clone!(state => move |_, ctx| gtk_draw(ctx, &state)));
let (wild_scroll, wild_tree, wild_css_provider, wild_renderer, wild_column) = let (wild_scroll, wild_tree, wild_css_provider, wild_renderer, wild_column) =
CmdLine::create_widlmenu(); CmdLine::create_widlmenu(&state);
content.pack_start(&wild_scroll, false, true, 0); content.pack_start(&wild_scroll, false, true, 0);
popover.add(&content); popover.add(&content);
@ -311,7 +314,9 @@ impl CmdLine {
} }
} }
fn create_widlmenu() -> ( fn create_widlmenu(
state: &Arc<UiMutex<State>>,
) -> (
gtk::ScrolledWindow, gtk::ScrolledWindow,
gtk::TreeView, gtk::TreeView,
gtk::CssProvider, gtk::CssProvider,
@ -342,11 +347,23 @@ impl CmdLine {
scroll.add(&tree); scroll.add(&tree);
tree.connect_button_press_event(clone!(state => move |tree, ev| {
let state = state.borrow();
let nvim = state.nvim.as_ref().unwrap().nvim();
if let Some(mut nvim) = nvim {
popup_menu::tree_button_press(tree, ev, &mut *nvim, "");
}
Inhibit(false)
}));
(scroll, tree, css_provider, renderer, column) (scroll, tree, css_provider, renderer, column)
} }
pub fn show_level(&mut self, ctx: &CmdLineContext) { pub fn show_level(&mut self, ctx: &CmdLineContext) {
let mut state = self.state.borrow_mut(); let mut state = self.state.borrow_mut();
if state.nvim.is_none() {
state.nvim = Some(ctx.nvim.clone());
}
let render_state = state.render_state.clone(); let render_state = state.render_state.clone();
let render_state = render_state.borrow(); let render_state = render_state.borrow();
@ -560,7 +577,8 @@ fn gtk_draw(ctx: &cairo::Context, state: &Arc<UiMutex<State>>) -> Inhibit {
Inhibit(false) Inhibit(false)
} }
pub struct CmdLineContext { pub struct CmdLineContext<'a> {
pub nvim: &'a Rc<NeovimClient>,
pub content: Vec<(HashMap<String, Value>, String)>, pub content: Vec<(HashMap<String, Value>, String)>,
pub pos: u64, pub pos: u64,
pub firstc: String, pub firstc: String,
@ -574,7 +592,7 @@ pub struct CmdLineContext {
pub max_width: i32, pub max_width: i32,
} }
impl CmdLineContext { impl<'a> CmdLineContext<'a> {
fn get_lines(&self) -> LineContent { fn get_lines(&self) -> LineContent {
let content_line: Vec<(Option<Attrs>, Vec<char>)> = self.content let content_line: Vec<(Option<Attrs>, Vec<char>)> = self.content
.iter() .iter()

View File

@ -1,6 +1,7 @@
use std::rc::Rc; use std::rc::Rc;
use std::cell::RefCell; use std::cell::RefCell;
use std::cmp::min; use std::cmp::min;
use std::iter;
use gtk; use gtk;
use gtk::prelude::*; use gtk::prelude::*;
@ -235,10 +236,9 @@ impl PopupMenu {
let state = state_ref.borrow(); let state = state_ref.borrow();
let nvim = state.nvim.as_ref().unwrap().nvim(); let nvim = state.nvim.as_ref().unwrap().nvim();
if let Some(mut nvim) = nvim { if let Some(mut nvim) = nvim {
tree_button_press(tree, ev, &mut *nvim) tree_button_press(tree, ev, &mut *nvim, "<C-y>");
} else {
Inhibit(false)
} }
Inhibit(false)
}); });
let state_ref = state.clone(); let state_ref = state.clone();
@ -302,9 +302,14 @@ pub struct PopupMenuContext<'a> {
pub max_width: i32, pub max_width: i32,
} }
fn tree_button_press(tree: &gtk::TreeView, ev: &EventButton, nvim: &mut Neovim) -> Inhibit { pub fn tree_button_press(
tree: &gtk::TreeView,
ev: &EventButton,
nvim: &mut Neovim,
last_command: &str,
) {
if ev.get_event_type() != EventType::ButtonPress { if ev.get_event_type() != EventType::ButtonPress {
return Inhibit(false); return;
} }
let (paths, ..) = tree.get_selection().get_selected_rows(); let (paths, ..) = tree.get_selection().get_selected_rows();
@ -325,21 +330,20 @@ fn tree_button_press(tree: &gtk::TreeView, ev: &EventButton, nvim: &mut Neovim)
let scroll_count = find_scroll_count(selected_idx, target_idx); let scroll_count = find_scroll_count(selected_idx, target_idx);
let mut apply_command = String::new(); let mut apply_command: String = if target_idx > selected_idx {
(0..scroll_count)
for _ in 0..scroll_count { .map(|_| "<C-n>")
if target_idx > selected_idx { .chain(iter::once(last_command))
apply_command.push_str("<C-n>"); .collect()
} else { } else {
apply_command.push_str("<C-p>"); (0..scroll_count)
} .map(|_| "<C-p>")
} .chain(iter::once(last_command))
apply_command.push_str("<C-y>"); .collect()
};
nvim.input(&apply_command).report_err(); nvim.input(&apply_command).report_err();
} }
Inhibit(false)
} }
fn find_scroll_count(selected_idx: i32, target_idx: i32) -> i32 { fn find_scroll_count(selected_idx: i32, target_idx: i32) -> i32 {

View File

@ -1364,6 +1364,7 @@ impl State {
let render_state = self.render_state.borrow(); let render_state = self.render_state.borrow();
let (x, y, width, height) = cursor.to_area(render_state.font_ctx.cell_metrics()); let (x, y, width, height) = cursor.to_area(render_state.font_ctx.cell_metrics());
let ctx = CmdLineContext { let ctx = CmdLineContext {
nvim: &self.nvim,
content, content,
pos, pos,
firstc, firstc,