mirror of
https://github.com/helix-editor/helix.git
synced 2025-01-18 21:17:08 +04:00
Fix coord mapping, add vertical move.
This commit is contained in:
parent
10d53f3ef0
commit
195aad4675
@ -69,6 +69,7 @@ pub fn move_pos(
|
||||
(Direction::Forward, Granularity::Character) => {
|
||||
nth_next_grapheme_boundary(&text.slice(..), pos, n)
|
||||
}
|
||||
(_, Granularity::Line) => move_vertically(&text.slice(..), dir, pos, n),
|
||||
_ => pos,
|
||||
}
|
||||
}
|
||||
@ -125,7 +126,8 @@ pub fn extend_selection(
|
||||
pub fn coords_at_pos(text: &RopeSlice, pos: usize) -> Coords {
|
||||
let line = text.char_to_line(pos);
|
||||
let line_start = text.line_to_char(line);
|
||||
let col = RopeGraphemes::new(&text.slice(line_start..pos)).count();
|
||||
// convert to 0-indexed
|
||||
let col = text.slice(line_start..pos).len_chars().saturating_sub(1);
|
||||
(line, col)
|
||||
}
|
||||
|
||||
@ -136,6 +138,31 @@ pub fn pos_at_coords(text: &RopeSlice, coords: Coords) -> usize {
|
||||
nth_next_grapheme_boundary(text, line_start, col)
|
||||
}
|
||||
|
||||
fn move_vertically(text: &RopeSlice, dir: Direction, pos: usize, n: usize) -> usize {
|
||||
let (line, col) = coords_at_pos(text, pos);
|
||||
|
||||
let new_line = match dir {
|
||||
Direction::Backward => line.saturating_sub(n),
|
||||
Direction::Forward => std::cmp::min(line.saturating_add(n), text.len_lines() - 1),
|
||||
};
|
||||
|
||||
// convert to 0-indexed
|
||||
let new_line_len = text.line(new_line).len_chars().saturating_sub(1);
|
||||
|
||||
let new_col = if new_line_len < col {
|
||||
// TODO: preserve horiz here
|
||||
new_line_len
|
||||
} else {
|
||||
col
|
||||
};
|
||||
|
||||
pos_at_coords(text, (new_line, new_col))
|
||||
}
|
||||
|
||||
/// A command is a function that takes the current state and a count, and does a side-effect on the
|
||||
/// state (usually by creating and applying a transaction).
|
||||
type Command = fn(state: &mut State, count: usize) -> bool;
|
||||
|
||||
#[cfg(test)]
|
||||
mod test {
|
||||
use super::*;
|
||||
@ -144,9 +171,9 @@ mod test {
|
||||
fn test_coords_at_pos() {
|
||||
let text = Rope::from("ḧëḷḷö\nẅöṛḷḋ");
|
||||
assert_eq!(coords_at_pos(&text.slice(..), 0), (0, 0));
|
||||
assert_eq!(coords_at_pos(&text.slice(..), 5), (0, 5)); // position on \n
|
||||
assert_eq!(coords_at_pos(&text.slice(..), 5), (0, 4)); // position on \n
|
||||
assert_eq!(coords_at_pos(&text.slice(..), 6), (1, 0)); // position on w
|
||||
assert_eq!(coords_at_pos(&text.slice(..), 11), (1, 5)); // position on d
|
||||
assert_eq!(coords_at_pos(&text.slice(..), 11), (1, 4)); // position on d
|
||||
}
|
||||
|
||||
#[test]
|
||||
@ -157,4 +184,16 @@ fn test_pos_at_coords() {
|
||||
assert_eq!(pos_at_coords(&text.slice(..), (1, 0)), 6); // position on w
|
||||
assert_eq!(pos_at_coords(&text.slice(..), (1, 5)), 11); // position on d
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_vertical_move() {
|
||||
let text = Rope::from("abcd\nefg\nwrs");
|
||||
let pos = pos_at_coords(&text.slice(..), (0, 4));
|
||||
let slice = text.slice(..);
|
||||
|
||||
assert_eq!(
|
||||
coords_at_pos(&slice, move_vertically(&slice, Direction::Forward, pos, 1)),
|
||||
(1, 2)
|
||||
);
|
||||
}
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user