mirror of
https://github.com/wez/wezterm.git
synced 2024-11-23 23:21:08 +03:00
wezterm: add mouseless-copy overlay
This allows navigating the scrollback and selecting text using only the keyboard.
This commit is contained in:
parent
1ec37da248
commit
29239c632b
428
src/frontend/gui/overlay/copy.rs
Normal file
428
src/frontend/gui/overlay/copy.rs
Normal file
@ -0,0 +1,428 @@
|
||||
use crate::frontend::gui::selection::{SelectionCoordinate, SelectionRange};
|
||||
use crate::frontend::gui::termwindow::TermWindow;
|
||||
use crate::mux::domain::DomainId;
|
||||
use crate::mux::renderable::*;
|
||||
use crate::mux::tab::{Tab, TabId};
|
||||
use portable_pty::PtySize;
|
||||
use rangeset::RangeSet;
|
||||
use std::cell::{RefCell, RefMut};
|
||||
use std::ops::Range;
|
||||
use std::rc::Rc;
|
||||
use std::sync::Arc;
|
||||
use term::color::ColorPalette;
|
||||
use term::{Clipboard, KeyCode, KeyModifiers, Line, MouseEvent, StableRowIndex};
|
||||
use url::Url;
|
||||
use window::WindowOps;
|
||||
|
||||
pub struct CopyOverlay {
|
||||
delegate: Rc<dyn Tab>,
|
||||
render: RefCell<CopyRenderable>,
|
||||
}
|
||||
|
||||
struct CopyRenderable {
|
||||
cursor: StableCursorPosition,
|
||||
delegate: Rc<dyn Tab>,
|
||||
start: Option<SelectionCoordinate>,
|
||||
viewport: Option<StableRowIndex>,
|
||||
/// We use this to cancel ourselves later
|
||||
window: ::window::Window,
|
||||
}
|
||||
|
||||
struct Dimensions {
|
||||
vertical_gap: isize,
|
||||
dims: RenderableDimensions,
|
||||
top: StableRowIndex,
|
||||
}
|
||||
|
||||
impl CopyOverlay {
|
||||
pub fn with_tab(term_window: &TermWindow, tab: &Rc<dyn Tab>) -> Rc<dyn Tab> {
|
||||
let mut cursor = tab.renderer().get_cursor_position();
|
||||
cursor.shape = termwiz::surface::CursorShape::SteadyBlock;
|
||||
|
||||
let window = term_window.window.clone().unwrap();
|
||||
let render = CopyRenderable {
|
||||
cursor,
|
||||
window,
|
||||
delegate: Rc::clone(tab),
|
||||
start: None,
|
||||
viewport: term_window.get_viewport(tab.tab_id()),
|
||||
};
|
||||
Rc::new(CopyOverlay {
|
||||
delegate: Rc::clone(tab),
|
||||
render: RefCell::new(render),
|
||||
})
|
||||
}
|
||||
|
||||
pub fn viewport_changed(&self, viewport: Option<StableRowIndex>) {
|
||||
let mut r = self.render.borrow_mut();
|
||||
r.viewport = viewport;
|
||||
}
|
||||
}
|
||||
|
||||
impl CopyRenderable {
|
||||
fn clamp_cursor_to_scrollback(&mut self) {
|
||||
let dims = self.delegate.renderer().get_dimensions();
|
||||
if self.cursor.x >= dims.cols {
|
||||
self.cursor.x = dims.cols - 1;
|
||||
}
|
||||
if self.cursor.y < dims.scrollback_top {
|
||||
self.cursor.y = dims.scrollback_top;
|
||||
}
|
||||
|
||||
let max_row = dims.scrollback_top + dims.scrollback_rows as isize;
|
||||
if self.cursor.y >= max_row {
|
||||
self.cursor.y = max_row - 1;
|
||||
}
|
||||
}
|
||||
|
||||
fn select_to_cursor_pos(&mut self) {
|
||||
self.clamp_cursor_to_scrollback();
|
||||
if let Some(start) = self.start {
|
||||
let start = SelectionCoordinate {
|
||||
x: start.x,
|
||||
y: start.y,
|
||||
};
|
||||
|
||||
let end = SelectionCoordinate {
|
||||
x: self.cursor.x,
|
||||
y: self.cursor.y,
|
||||
};
|
||||
|
||||
self.adjust_selection(start, SelectionRange { start, end });
|
||||
} else {
|
||||
self.adjust_viewport_for_cursor_position();
|
||||
self.window.invalidate();
|
||||
}
|
||||
}
|
||||
|
||||
fn adjust_selection(&self, start: SelectionCoordinate, range: SelectionRange) {
|
||||
let tab_id = self.delegate.tab_id();
|
||||
self.window.apply(move |term_window, window| {
|
||||
if let Some(term_window) = term_window.downcast_mut::<TermWindow>() {
|
||||
let mut selection = term_window.selection(tab_id);
|
||||
selection.start = Some(start);
|
||||
selection.range = Some(range);
|
||||
window.invalidate();
|
||||
}
|
||||
Ok(())
|
||||
});
|
||||
self.adjust_viewport_for_cursor_position();
|
||||
}
|
||||
|
||||
fn dimensions(&self) -> Dimensions {
|
||||
const VERTICAL_GAP: isize = 5;
|
||||
let dims = self.delegate.renderer().get_dimensions();
|
||||
let vertical_gap = if dims.physical_top <= VERTICAL_GAP {
|
||||
1
|
||||
} else {
|
||||
VERTICAL_GAP
|
||||
};
|
||||
let top = self.viewport.unwrap_or_else(|| dims.physical_top);
|
||||
Dimensions {
|
||||
vertical_gap,
|
||||
top,
|
||||
dims,
|
||||
}
|
||||
}
|
||||
|
||||
fn adjust_viewport_for_cursor_position(&self) {
|
||||
let dims = self.dimensions();
|
||||
|
||||
if dims.top > self.cursor.y {
|
||||
// Cursor is off the top of the viewport; adjust
|
||||
self.set_viewport(Some(self.cursor.y.saturating_sub(dims.vertical_gap)));
|
||||
return;
|
||||
}
|
||||
|
||||
let top_gap = self.cursor.y - dims.top;
|
||||
if top_gap < dims.vertical_gap {
|
||||
// Increase the gap so we can "look ahead"
|
||||
self.set_viewport(Some(self.cursor.y.saturating_sub(dims.vertical_gap)));
|
||||
return;
|
||||
}
|
||||
|
||||
let bottom_gap = (dims.dims.viewport_rows as isize).saturating_sub(top_gap);
|
||||
if bottom_gap < dims.vertical_gap {
|
||||
self.set_viewport(Some(dims.top + dims.vertical_gap - bottom_gap));
|
||||
}
|
||||
}
|
||||
|
||||
fn set_viewport(&self, row: Option<StableRowIndex>) {
|
||||
let dims = self.delegate.renderer().get_dimensions();
|
||||
let tab_id = self.delegate.tab_id();
|
||||
self.window.apply(move |term_window, _window| {
|
||||
if let Some(term_window) = term_window.downcast_mut::<TermWindow>() {
|
||||
term_window.set_viewport(tab_id, row, dims);
|
||||
}
|
||||
Ok(())
|
||||
});
|
||||
}
|
||||
|
||||
fn close(&self) {
|
||||
self.set_viewport(None);
|
||||
TermWindow::schedule_cancel_overlay(self.window.clone(), self.delegate.tab_id());
|
||||
}
|
||||
|
||||
fn page_up(&mut self) {
|
||||
let dims = self.dimensions();
|
||||
self.cursor.y -= dims.dims.viewport_rows as isize;
|
||||
self.select_to_cursor_pos();
|
||||
}
|
||||
|
||||
fn page_down(&mut self) {
|
||||
let dims = self.dimensions();
|
||||
self.cursor.y += dims.dims.viewport_rows as isize;
|
||||
self.select_to_cursor_pos();
|
||||
}
|
||||
|
||||
fn move_to_viewport_middle(&mut self) {
|
||||
let dims = self.dimensions();
|
||||
self.cursor.y = dims.top + (dims.dims.viewport_rows as isize) / 2;
|
||||
self.select_to_cursor_pos();
|
||||
}
|
||||
|
||||
fn move_to_viewport_top(&mut self) {
|
||||
let dims = self.dimensions();
|
||||
self.cursor.y = dims.top + dims.vertical_gap;
|
||||
self.select_to_cursor_pos();
|
||||
}
|
||||
|
||||
fn move_to_viewport_bottom(&mut self) {
|
||||
let dims = self.dimensions();
|
||||
self.cursor.y = dims.top + (dims.dims.viewport_rows as isize) - dims.vertical_gap;
|
||||
self.select_to_cursor_pos();
|
||||
}
|
||||
|
||||
fn move_left_single_cell(&mut self) {
|
||||
self.cursor.x = self.cursor.x.saturating_sub(1);
|
||||
self.select_to_cursor_pos();
|
||||
}
|
||||
|
||||
fn move_right_single_cell(&mut self) {
|
||||
self.cursor.x += 1;
|
||||
self.select_to_cursor_pos();
|
||||
}
|
||||
|
||||
fn move_up_single_row(&mut self) {
|
||||
self.cursor.y = self.cursor.y.saturating_sub(1);
|
||||
self.select_to_cursor_pos();
|
||||
}
|
||||
|
||||
fn move_down_single_row(&mut self) {
|
||||
self.cursor.y += 1;
|
||||
self.select_to_cursor_pos();
|
||||
}
|
||||
fn move_to_start_of_line(&mut self) {
|
||||
self.cursor.x = 0;
|
||||
self.select_to_cursor_pos();
|
||||
}
|
||||
|
||||
fn move_to_start_of_next_line(&mut self) {
|
||||
self.cursor.x = 0;
|
||||
self.cursor.y += 1;
|
||||
self.select_to_cursor_pos();
|
||||
}
|
||||
|
||||
fn move_to_top(&mut self) {
|
||||
// This will get fixed up by clamp_cursor_to_scrollback
|
||||
self.cursor.y = 0;
|
||||
self.select_to_cursor_pos();
|
||||
}
|
||||
|
||||
fn move_to_bottom(&mut self) {
|
||||
// This will get fixed up by clamp_cursor_to_scrollback
|
||||
self.cursor.y = isize::max_value();
|
||||
self.select_to_cursor_pos();
|
||||
}
|
||||
|
||||
fn move_to_end_of_line_content(&mut self) {
|
||||
let y = self.cursor.y;
|
||||
let (top, lines) = self.get_lines(y..y + 1);
|
||||
if let Some(line) = lines.get(0) {
|
||||
self.cursor.y = top;
|
||||
self.cursor.x = 0;
|
||||
for (x, cell) in line.cells().iter().enumerate().rev() {
|
||||
if cell.str() != " " {
|
||||
self.cursor.x = x;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
self.select_to_cursor_pos();
|
||||
}
|
||||
|
||||
fn move_to_start_of_line_content(&mut self) {
|
||||
let y = self.cursor.y;
|
||||
let (top, lines) = self.get_lines(y..y + 1);
|
||||
if let Some(line) = lines.get(0) {
|
||||
self.cursor.y = top;
|
||||
self.cursor.x = 0;
|
||||
for (x, cell) in line.cells().iter().enumerate() {
|
||||
if cell.str() != " " {
|
||||
self.cursor.x = x;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
self.select_to_cursor_pos();
|
||||
}
|
||||
|
||||
fn toggle_selection_by_cell(&mut self) {
|
||||
if self.start.take().is_none() {
|
||||
let coord = SelectionCoordinate {
|
||||
x: self.cursor.x,
|
||||
y: self.cursor.y,
|
||||
};
|
||||
self.start.replace(coord);
|
||||
self.select_to_cursor_pos();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl Tab for CopyOverlay {
|
||||
fn tab_id(&self) -> TabId {
|
||||
self.delegate.tab_id()
|
||||
}
|
||||
|
||||
fn renderer(&self) -> RefMut<dyn Renderable> {
|
||||
self.render.borrow_mut()
|
||||
}
|
||||
|
||||
fn get_title(&self) -> String {
|
||||
format!("Copy mode: {}", self.delegate.get_title())
|
||||
}
|
||||
|
||||
fn send_paste(&self, _text: &str) -> anyhow::Result<()> {
|
||||
anyhow::bail!("ignoring paste while copying");
|
||||
}
|
||||
|
||||
fn reader(&self) -> anyhow::Result<Box<dyn std::io::Read + Send>> {
|
||||
panic!("do not call reader on CopyOverlay bar tab instance");
|
||||
}
|
||||
|
||||
fn writer(&self) -> RefMut<dyn std::io::Write> {
|
||||
self.delegate.writer()
|
||||
}
|
||||
|
||||
fn resize(&self, size: PtySize) -> anyhow::Result<()> {
|
||||
self.delegate.resize(size)
|
||||
}
|
||||
|
||||
fn key_down(&self, key: KeyCode, mods: KeyModifiers) -> anyhow::Result<()> {
|
||||
match (key, mods) {
|
||||
(KeyCode::Char('c'), KeyModifiers::CTRL)
|
||||
| (KeyCode::Char('g'), KeyModifiers::CTRL)
|
||||
| (KeyCode::Char('q'), KeyModifiers::NONE)
|
||||
| (KeyCode::Escape, KeyModifiers::NONE) => self.render.borrow().close(),
|
||||
(KeyCode::Char('h'), KeyModifiers::NONE) | (KeyCode::LeftArrow, KeyModifiers::NONE) => {
|
||||
self.render.borrow_mut().move_left_single_cell();
|
||||
}
|
||||
(KeyCode::Char('j'), KeyModifiers::NONE) | (KeyCode::DownArrow, KeyModifiers::NONE) => {
|
||||
self.render.borrow_mut().move_down_single_row();
|
||||
}
|
||||
(KeyCode::Char('k'), KeyModifiers::NONE) | (KeyCode::UpArrow, KeyModifiers::NONE) => {
|
||||
self.render.borrow_mut().move_up_single_row();
|
||||
}
|
||||
(KeyCode::Char('l'), KeyModifiers::NONE)
|
||||
| (KeyCode::RightArrow, KeyModifiers::NONE) => {
|
||||
self.render.borrow_mut().move_right_single_cell();
|
||||
}
|
||||
(KeyCode::Char('0'), KeyModifiers::NONE) => {
|
||||
self.render.borrow_mut().move_to_start_of_line();
|
||||
}
|
||||
(KeyCode::Enter, KeyModifiers::NONE) => {
|
||||
self.render.borrow_mut().move_to_start_of_next_line();
|
||||
}
|
||||
(KeyCode::Char('$'), KeyModifiers::SHIFT) | // FIXME: normalize the shift away!
|
||||
(KeyCode::Char('$'), KeyModifiers::NONE) => {
|
||||
self.render.borrow_mut().move_to_end_of_line_content();
|
||||
}
|
||||
(KeyCode::Char('m'), KeyModifiers::ALT) |
|
||||
(KeyCode::Char('^'), KeyModifiers::SHIFT) | // FIXME: normalize the shift away!
|
||||
(KeyCode::Char('^'), KeyModifiers::NONE) => {
|
||||
self.render.borrow_mut().move_to_start_of_line_content();
|
||||
}
|
||||
(KeyCode::Char(' '), KeyModifiers::NONE) | (KeyCode::Char('v'), KeyModifiers::NONE) => {
|
||||
self.render.borrow_mut().toggle_selection_by_cell();
|
||||
}
|
||||
(KeyCode::Char('G'), KeyModifiers::SHIFT) | // FIXME: normalize the shift away!
|
||||
(KeyCode::Char('G'), KeyModifiers::NONE) => {
|
||||
self.render.borrow_mut().move_to_bottom();
|
||||
}
|
||||
(KeyCode::Char('g'), KeyModifiers::NONE) => {
|
||||
self.render.borrow_mut().move_to_top();
|
||||
}
|
||||
(KeyCode::Char('H'), KeyModifiers::SHIFT) | // FIXME: normalize the shift away!
|
||||
(KeyCode::Char('H'), KeyModifiers::NONE) => {
|
||||
self.render.borrow_mut().move_to_viewport_top();
|
||||
}
|
||||
(KeyCode::Char('M'), KeyModifiers::SHIFT) | // FIXME: normalize the shift away!
|
||||
(KeyCode::Char('M'), KeyModifiers::NONE) => {
|
||||
self.render.borrow_mut().move_to_viewport_middle();
|
||||
}
|
||||
(KeyCode::Char('L'), KeyModifiers::SHIFT) | // FIXME: normalize the shift away!
|
||||
(KeyCode::Char('L'), KeyModifiers::NONE) => {
|
||||
self.render.borrow_mut().move_to_viewport_bottom();
|
||||
}
|
||||
(KeyCode::PageUp, KeyModifiers::NONE) | (KeyCode::Char('b'), KeyModifiers::CTRL) => self.render.borrow_mut().page_up(),
|
||||
(KeyCode::PageDown, KeyModifiers::NONE) | (KeyCode::Char('f'), KeyModifiers::CTRL) => self.render.borrow_mut().page_down(),
|
||||
_ => {}
|
||||
}
|
||||
Ok(())
|
||||
}
|
||||
|
||||
fn mouse_event(&self, _event: MouseEvent) -> anyhow::Result<()> {
|
||||
anyhow::bail!("ignoring mouse while copying");
|
||||
}
|
||||
|
||||
fn advance_bytes(&self, buf: &[u8]) {
|
||||
self.delegate.advance_bytes(buf)
|
||||
}
|
||||
|
||||
fn is_dead(&self) -> bool {
|
||||
self.delegate.is_dead()
|
||||
}
|
||||
|
||||
fn palette(&self) -> ColorPalette {
|
||||
self.delegate.palette()
|
||||
}
|
||||
|
||||
fn domain_id(&self) -> DomainId {
|
||||
self.delegate.domain_id()
|
||||
}
|
||||
|
||||
fn erase_scrollback(&self) {
|
||||
self.delegate.erase_scrollback()
|
||||
}
|
||||
|
||||
fn is_mouse_grabbed(&self) -> bool {
|
||||
// Force grabbing off while we're searching
|
||||
false
|
||||
}
|
||||
|
||||
fn set_clipboard(&self, clipboard: &Arc<dyn Clipboard>) {
|
||||
self.delegate.set_clipboard(clipboard)
|
||||
}
|
||||
|
||||
fn get_current_working_dir(&self) -> Option<Url> {
|
||||
self.delegate.get_current_working_dir()
|
||||
}
|
||||
}
|
||||
|
||||
impl Renderable for CopyRenderable {
|
||||
fn get_cursor_position(&self) -> StableCursorPosition {
|
||||
self.cursor
|
||||
}
|
||||
|
||||
fn get_dirty_lines(&self, lines: Range<StableRowIndex>) -> RangeSet<StableRowIndex> {
|
||||
self.delegate.renderer().get_dirty_lines(lines)
|
||||
}
|
||||
|
||||
fn get_lines(&mut self, lines: Range<StableRowIndex>) -> (StableRowIndex, Vec<Line>) {
|
||||
self.delegate.renderer().get_lines(lines)
|
||||
}
|
||||
|
||||
fn get_dimensions(&self) -> RenderableDimensions {
|
||||
self.delegate.renderer().get_dimensions()
|
||||
}
|
||||
}
|
@ -4,10 +4,12 @@ use crate::termwiztermtab::{allocate, TermWizTerminal};
|
||||
use std::pin::Pin;
|
||||
use std::rc::Rc;
|
||||
|
||||
mod copy;
|
||||
mod launcher;
|
||||
mod search;
|
||||
mod tabnavigator;
|
||||
|
||||
pub use copy::CopyOverlay;
|
||||
pub use launcher::launcher;
|
||||
pub use search::SearchOverlay;
|
||||
pub use tabnavigator::tab_navigator;
|
||||
|
@ -8,7 +8,9 @@ use crate::font::units::*;
|
||||
use crate::font::FontConfiguration;
|
||||
use crate::frontend::activity::Activity;
|
||||
use crate::frontend::front_end;
|
||||
use crate::frontend::gui::overlay::{launcher, start_overlay, tab_navigator, SearchOverlay};
|
||||
use crate::frontend::gui::overlay::{
|
||||
launcher, start_overlay, tab_navigator, CopyOverlay, SearchOverlay,
|
||||
};
|
||||
use crate::frontend::gui::scrollbar::*;
|
||||
use crate::frontend::gui::selection::*;
|
||||
use crate::frontend::gui::tabbar::{TabBarItem, TabBarState};
|
||||
@ -732,7 +734,9 @@ impl TermWindow {
|
||||
let dirty = render.get_dirty_lines(visible_range);
|
||||
|
||||
if !dirty.is_empty() {
|
||||
if tab.downcast_ref::<SearchOverlay>().is_none() {
|
||||
if tab.downcast_ref::<SearchOverlay>().is_none()
|
||||
&& tab.downcast_ref::<CopyOverlay>().is_none()
|
||||
{
|
||||
// If any of the changed lines intersect with the
|
||||
// selection, then we need to clear the selection, but not
|
||||
// when the search overlay is active; the search overlay
|
||||
@ -1032,7 +1036,12 @@ impl TermWindow {
|
||||
let tab_no = window.get_active_idx();
|
||||
|
||||
let title = match window.get_active() {
|
||||
Some(tab) => tab.get_title(),
|
||||
Some(tab) => self
|
||||
.tab_state(tab.tab_id())
|
||||
.overlay
|
||||
.as_ref()
|
||||
.unwrap_or(tab)
|
||||
.get_title(),
|
||||
None => return,
|
||||
};
|
||||
|
||||
@ -1580,6 +1589,10 @@ impl TermWindow {
|
||||
let search = SearchOverlay::with_tab(self, tab, pattern.clone());
|
||||
self.assign_overlay(tab.tab_id(), search);
|
||||
}
|
||||
ActivateCopyMode => {
|
||||
let copy = CopyOverlay::with_tab(self, tab);
|
||||
self.assign_overlay(tab.tab_id(), copy);
|
||||
}
|
||||
};
|
||||
Ok(())
|
||||
}
|
||||
@ -2793,6 +2806,8 @@ impl TermWindow {
|
||||
if let Some(overlay) = state.overlay.as_ref() {
|
||||
if let Some(search_overlay) = overlay.downcast_ref::<SearchOverlay>() {
|
||||
search_overlay.viewport_changed(pos);
|
||||
} else if let Some(copy) = overlay.downcast_ref::<CopyOverlay>() {
|
||||
copy.viewport_changed(pos);
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -3161,8 +3176,9 @@ impl TermWindow {
|
||||
});
|
||||
}
|
||||
|
||||
pub fn assign_overlay(&self, tab_id: TabId, overlay: Rc<dyn Tab>) {
|
||||
pub fn assign_overlay(&mut self, tab_id: TabId, overlay: Rc<dyn Tab>) {
|
||||
self.tab_state(tab_id).overlay.replace(overlay);
|
||||
self.update_title();
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -101,6 +101,7 @@ pub enum KeyAssignment {
|
||||
ShowLauncher,
|
||||
ClearScrollback,
|
||||
Search(Pattern),
|
||||
ActivateCopyMode,
|
||||
|
||||
SelectTextAtMouseCursor(SelectionMode),
|
||||
ExtendSelectionToMouseCursor(Option<SelectionMode>),
|
||||
@ -245,6 +246,7 @@ impl InputMap {
|
||||
[KeyModifiers::SHIFT, KeyCode::PageUp, ScrollByPage(-1)],
|
||||
[KeyModifiers::SHIFT, KeyCode::PageDown, ScrollByPage(1)],
|
||||
[KeyModifiers::ALT, KeyCode::Char('9'), ShowTabNavigator],
|
||||
[ctrl_shift, KeyCode::Char('X'), ActivateCopyMode],
|
||||
);
|
||||
|
||||
#[cfg(target_os = "macos")]
|
||||
|
Loading…
Reference in New Issue
Block a user