mirror of
https://github.com/ilyakooo0/helix.git
synced 2024-11-28 12:42:09 +03:00
Refactor and add tests for surround
This commit is contained in:
parent
4754b2e5ae
commit
86271bac18
@ -13,6 +13,14 @@ pub const PAIRS: &[(char, char)] = &[
|
|||||||
|
|
||||||
/// Given any char in [PAIRS], return the open and closing chars. If not found in
|
/// Given any char in [PAIRS], return the open and closing chars. If not found in
|
||||||
/// [PAIRS] return (ch, ch).
|
/// [PAIRS] return (ch, ch).
|
||||||
|
///
|
||||||
|
/// ```
|
||||||
|
/// use helix_core::surround::get_pair;
|
||||||
|
///
|
||||||
|
/// assert_eq!(get_pair('['), ('[', ']'));
|
||||||
|
/// assert_eq!(get_pair('}'), ('{', '}'));
|
||||||
|
/// assert_eq!(get_pair('"'), ('"', '"'));
|
||||||
|
/// ```
|
||||||
pub fn get_pair(ch: char) -> (char, char) {
|
pub fn get_pair(ch: char) -> (char, char) {
|
||||||
PAIRS
|
PAIRS
|
||||||
.iter()
|
.iter()
|
||||||
@ -50,17 +58,101 @@ pub fn get_surround_pos(
|
|||||||
let mut change_pos = Vec::new();
|
let mut change_pos = Vec::new();
|
||||||
|
|
||||||
for range in selection {
|
for range in selection {
|
||||||
let head = range.head;
|
let (open_pos, close_pos) = find_nth_pairs_pos(text, ch, range.head, skip)?;
|
||||||
|
if change_pos.contains(&open_pos) || change_pos.contains(&close_pos) {
|
||||||
match find_nth_pairs_pos(text, ch, head, skip) {
|
return None;
|
||||||
Some((open_pos, close_pos)) => {
|
|
||||||
if change_pos.contains(&open_pos) || change_pos.contains(&close_pos) {
|
|
||||||
return None;
|
|
||||||
}
|
|
||||||
change_pos.extend_from_slice(&[open_pos, close_pos]);
|
|
||||||
}
|
|
||||||
None => return None,
|
|
||||||
}
|
}
|
||||||
|
change_pos.extend_from_slice(&[open_pos, close_pos]);
|
||||||
}
|
}
|
||||||
Some(change_pos)
|
Some(change_pos)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod test {
|
||||||
|
use super::*;
|
||||||
|
use crate::Range;
|
||||||
|
|
||||||
|
use ropey::Rope;
|
||||||
|
use smallvec::SmallVec;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_find_nth_pairs_pos() {
|
||||||
|
let doc = Rope::from("some (text) here");
|
||||||
|
let slice = doc.slice(..);
|
||||||
|
|
||||||
|
// cursor on [t]ext
|
||||||
|
assert_eq!(find_nth_pairs_pos(slice, '(', 6, 1), Some((5, 10)));
|
||||||
|
assert_eq!(find_nth_pairs_pos(slice, ')', 6, 1), Some((5, 10)));
|
||||||
|
// cursor on so[m]e
|
||||||
|
assert_eq!(find_nth_pairs_pos(slice, '(', 2, 1), None);
|
||||||
|
// cursor on bracket itself
|
||||||
|
// assert_eq!(find_nth_pairs_pos(slice, '(', 5, 1), Some((5, 10)));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_find_nth_pairs_pos_skip() {
|
||||||
|
let doc = Rope::from("(so (many (good) text) here)");
|
||||||
|
let slice = doc.slice(..);
|
||||||
|
|
||||||
|
// cursor on go[o]d
|
||||||
|
assert_eq!(find_nth_pairs_pos(slice, '(', 13, 1), Some((10, 15)));
|
||||||
|
assert_eq!(find_nth_pairs_pos(slice, '(', 13, 2), Some((4, 21)));
|
||||||
|
assert_eq!(find_nth_pairs_pos(slice, '(', 13, 3), Some((0, 27)));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_find_nth_pairs_pos_mixed() {
|
||||||
|
let doc = Rope::from("(so [many {good} text] here)");
|
||||||
|
let slice = doc.slice(..);
|
||||||
|
|
||||||
|
// cursor on go[o]d
|
||||||
|
assert_eq!(find_nth_pairs_pos(slice, '{', 13, 1), Some((10, 15)));
|
||||||
|
assert_eq!(find_nth_pairs_pos(slice, '[', 13, 1), Some((4, 21)));
|
||||||
|
assert_eq!(find_nth_pairs_pos(slice, '(', 13, 1), Some((0, 27)));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_get_surround_pos() {
|
||||||
|
let doc = Rope::from("(some) (chars)\n(newline)");
|
||||||
|
let slice = doc.slice(..);
|
||||||
|
let selection = Selection::new(
|
||||||
|
SmallVec::from_slice(&[Range::point(2), Range::point(9), Range::point(20)]),
|
||||||
|
0,
|
||||||
|
);
|
||||||
|
|
||||||
|
// cursor on s[o]me, c[h]ars, newl[i]ne
|
||||||
|
assert_eq!(
|
||||||
|
get_surround_pos(slice, &selection, '(', 1)
|
||||||
|
.unwrap()
|
||||||
|
.as_slice(),
|
||||||
|
&[0, 5, 7, 13, 15, 23]
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_get_surround_pos_bail() {
|
||||||
|
let doc = Rope::from("[some]\n(chars)xx\n(newline)");
|
||||||
|
let slice = doc.slice(..);
|
||||||
|
|
||||||
|
let selection = Selection::new(
|
||||||
|
SmallVec::from_slice(&[Range::point(2), Range::point(9)]),
|
||||||
|
0,
|
||||||
|
);
|
||||||
|
|
||||||
|
// cursor on s[o]me, c[h]ars
|
||||||
|
assert_eq!(
|
||||||
|
get_surround_pos(slice, &selection, '(', 1),
|
||||||
|
None // different surround chars
|
||||||
|
);
|
||||||
|
|
||||||
|
let selection = Selection::new(
|
||||||
|
SmallVec::from_slice(&[Range::point(14), Range::point(24)]),
|
||||||
|
0,
|
||||||
|
);
|
||||||
|
// cursor on [x]x, newli[n]e
|
||||||
|
assert_eq!(
|
||||||
|
get_surround_pos(slice, &selection, '(', 1),
|
||||||
|
None // overlapping surround chars
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user