summaryrefslogtreecommitdiff
path: root/alacritty_terminal/src/term/mod.rs
diff options
context:
space:
mode:
Diffstat (limited to 'alacritty_terminal/src/term/mod.rs')
-rw-r--r--alacritty_terminal/src/term/mod.rs91
1 files changed, 44 insertions, 47 deletions
diff --git a/alacritty_terminal/src/term/mod.rs b/alacritty_terminal/src/term/mod.rs
index ffb4a1e8..48b62d6b 100644
--- a/alacritty_terminal/src/term/mod.rs
+++ b/alacritty_terminal/src/term/mod.rs
@@ -73,6 +73,20 @@ impl Default for TermMode {
}
}
+/// Convert a terminal point to a viewport relative point.
+#[inline]
+pub fn point_to_viewport(display_offset: usize, point: Point) -> Option<Point<usize>> {
+ let viewport_line = point.line.0 + display_offset as i32;
+ usize::try_from(viewport_line).ok().map(|line| Point::new(line, point.column))
+}
+
+/// Convert a viewport relative point to a terminal point.
+#[inline]
+pub fn viewport_to_point(display_offset: usize, point: Point<usize>) -> Point {
+ let line = Line(point.line as i32) - display_offset;
+ Point::new(line, point.column)
+}
+
#[derive(Clone, Copy, Debug, PartialEq, Eq)]
pub struct LineDamageBounds {
/// Damaged line number.
@@ -149,6 +163,9 @@ struct TermDamageState {
/// Old terminal cursor point.
last_cursor: Point,
+ /// Last Vi cursor point.
+ last_vi_cursor_point: Option<Point<usize>>,
+
/// Old selection range.
last_selection: Option<SelectionRange>,
}
@@ -162,6 +179,7 @@ impl TermDamageState {
is_fully_damaged: true,
lines,
last_cursor: Default::default(),
+ last_vi_cursor_point: Default::default(),
last_selection: Default::default(),
}
}
@@ -170,6 +188,7 @@ impl TermDamageState {
fn resize(&mut self, num_cols: usize, num_lines: usize) {
// Reset point, so old cursor won't end up outside of the viewport.
self.last_cursor = Default::default();
+ self.last_vi_cursor_point = None;
self.last_selection = None;
self.is_fully_damaged = true;
@@ -373,13 +392,23 @@ impl<T> Term<T> {
self.damage_cursor();
self.damage.last_cursor = self.grid.cursor.point;
+ // Vi mode doesn't update the terminal content, thus only last vi cursor position and the
+ // new one should be damaged.
+ if let Some(last_vi_cursor_point) = self.damage.last_vi_cursor_point.take() {
+ self.damage.damage_point(last_vi_cursor_point)
+ }
+
+ let display_offset = self.grid().display_offset();
+
// Damage Vi cursor if it's present.
if self.mode.contains(TermMode::VI) {
- self.damage_vi_cursor();
+ let vi_cursor_point =
+ point_to_viewport(display_offset, self.vi_mode_cursor.point).unwrap();
+ self.damage.last_vi_cursor_point = Some(vi_cursor_point);
+ self.damage.damage_point(vi_cursor_point);
}
if self.damage.last_selection != selection {
- let display_offset = self.grid().display_offset();
for selection in self.damage.last_selection.into_iter().chain(selection) {
self.damage.damage_selection(selection, display_offset, self.columns());
}
@@ -749,9 +778,7 @@ impl<T> Term<T> {
}
// Move cursor.
- self.damage_vi_cursor();
self.vi_mode_cursor = self.vi_mode_cursor.motion(self, motion);
- self.damage_vi_cursor();
self.vi_mode_recompute_selection();
}
@@ -761,7 +788,6 @@ impl<T> Term<T> {
where
T: EventListener,
{
- self.damage_vi_cursor();
// Move viewport to make point visible.
self.scroll_to_point(point);
@@ -769,7 +795,6 @@ impl<T> Term<T> {
self.vi_mode_cursor.point = point;
self.vi_mode_recompute_selection();
- self.damage_vi_cursor();
}
/// Update the active selection to match the vi mode cursor position.
@@ -926,15 +951,6 @@ impl<T> Term<T> {
Point::new(self.grid.cursor.point.line.0 as usize, self.grid.cursor.point.column);
self.damage.damage_point(point);
}
-
- /// Damage `Vi` mode cursor.
- #[inline]
- pub fn damage_vi_cursor(&mut self) {
- let line = (self.grid.display_offset() as i32 + self.vi_mode_cursor.point.line.0)
- .clamp(0, self.screen_lines() as i32 - 1) as usize;
- let vi_point = Point::new(line, self.vi_mode_cursor.point.column);
- self.damage.damage_point(vi_point);
- }
}
impl<T> Dimensions for Term<T> {
@@ -2698,6 +2714,19 @@ mod tests {
let line = vi_cursor_point.line.0 as usize;
let left = vi_cursor_point.column.0 as usize;
let right = left;
+
+ let mut damaged_lines = match term.damage(None) {
+ TermDamage::Full => panic!("Expected partial damage, however got Full"),
+ TermDamage::Partial(damaged_lines) => damaged_lines,
+ };
+ // Skip cursor damage information, since we're just testing Vi cursor.
+ damaged_lines.next();
+ assert_eq!(damaged_lines.next(), Some(LineDamageBounds { line, left, right }));
+ assert_eq!(damaged_lines.next(), None);
+
+ // Ensure that old Vi cursor got damaged as well.
+ term.reset_damage();
+ term.toggle_vi_mode();
let mut damaged_lines = match term.damage(None) {
TermDamage::Full => panic!("Expected partial damage, however got Full"),
TermDamage::Partial(damaged_lines) => damaged_lines,
@@ -2807,38 +2836,6 @@ mod tests {
}
#[test]
- fn damage_vi_movements() {
- let size = TermSize::new(10, 10);
- let mut term = Term::new(&Config::default(), &size, ());
- let num_cols = term.columns();
- // Reset terminal for partial damage tests since it's initialized as fully damaged.
- term.reset_damage();
-
- // Enable Vi mode.
- term.toggle_vi_mode();
-
- // NOTE While we can use `[Term::damage]` to access terminal damage information, in the
- // following tests we will be accessing `term.damage.lines` directly to avoid adding extra
- // damage information (like cursor and Vi cursor), which we're not testing.
-
- term.vi_goto_point(Point::new(Line(5), Column(5)));
- assert_eq!(term.damage.lines[0], LineDamageBounds { line: 0, left: 0, right: 0 });
- assert_eq!(term.damage.lines[5], LineDamageBounds { line: 5, left: 5, right: 5 });
- term.damage.reset(num_cols);
-
- term.vi_motion(ViMotion::Up);
- term.vi_motion(ViMotion::Right);
- term.vi_motion(ViMotion::Up);
- term.vi_motion(ViMotion::Left);
- assert_eq!(term.damage.lines[3], LineDamageBounds { line: 3, left: 5, right: 6 });
- assert_eq!(term.damage.lines[4], LineDamageBounds { line: 4, left: 5, right: 6 });
- assert_eq!(term.damage.lines[5], LineDamageBounds { line: 5, left: 5, right: 5 });
-
- // Ensure that we haven't damaged entire terminal during the test.
- assert!(!term.damage.is_fully_damaged);
- }
-
- #[test]
fn full_damage() {
let size = TermSize::new(100, 10);
let mut term = Term::new(&Config::default(), &size, ());