generalize folding tests

By using xml-like tags, we will be able to test nested foldings.
This commit is contained in:
Aleksey Kladov 2018-12-20 22:30:30 +03:00
parent 61dcaa6add
commit 8d7e8a175e
2 changed files with 52 additions and 28 deletions

View file

@ -170,7 +170,7 @@ mod tests {
use test_utils::extract_ranges;
fn do_check(text: &str, fold_kinds: &[FoldKind]) {
let (ranges, text) = extract_ranges(text);
let (ranges, text) = extract_ranges(text, "fold");
let file = SourceFileNode::parse(&text);
let folds = folding_ranges(&file);
@ -198,22 +198,22 @@ mod tests {
#[test]
fn test_fold_comments() {
let text = r#"
<|>// Hello
<fold>// Hello
// this is a multiline
// comment
//<|>
//</fold>
// But this is not
fn main() {
<|>// We should
<fold>// We should
// also
// fold
// this one.<|>
<|>//! But this one is different
//! because it has another flavor<|>
<|>/* As does this
multiline comment */<|>
// this one.</fold>
<fold>//! But this one is different
//! because it has another flavor</fold>
<fold>/* As does this
multiline comment */</fold>
}"#;
let fold_kinds = &[
@ -228,11 +228,11 @@ fn main() {
#[test]
fn test_fold_imports() {
let text = r#"
<|>use std::{
<fold>use std::{
str,
vec,
io as iop
};<|>
};</fold>
fn main() {
}"#;
@ -244,12 +244,12 @@ fn main() {
#[test]
fn test_fold_import_groups() {
let text = r#"
<|>use std::str;
<fold>use std::str;
use std::vec;
use std::io as iop;<|>
use std::io as iop;</fold>
<|>use std::mem;
use std::f64;<|>
<fold>use std::mem;
use std::f64;</fold>
use std::collections::HashMap;
// Some random comment
@ -265,17 +265,17 @@ fn main() {
#[test]
fn test_fold_import_and_groups() {
let text = r#"
<|>use std::str;
<fold>use std::str;
use std::vec;
use std::io as iop;<|>
use std::io as iop;</fold>
<|>use std::mem;
use std::f64;<|>
<fold>use std::mem;
use std::f64;</fold>
<|>use std::collections::{
<fold>use std::collections::{
HashMap,
VecDeque,
};<|>
};</fold>
// Some random comment
fn main() {

View file

@ -66,15 +66,39 @@ pub fn try_extract_range(text: &str) -> Option<(TextRange, String)> {
Some((TextRange::from_to(start, end), text))
}
pub fn extract_ranges(text: &str) -> (Vec<TextRange>, String) {
/// Extracts ranges, marked with `<tag> </tag>` paris from the `text`
pub fn extract_ranges(mut text: &str, tag: &str) -> (Vec<TextRange>, String) {
let open = format!("<{}>", tag);
let close = format!("</{}>", tag);
let mut ranges = Vec::new();
let mut text = String::from(text);
while let Some((range, new_text)) = try_extract_range(&text) {
text = new_text;
ranges.push(range);
let mut res = String::new();
let mut stack = Vec::new();
loop {
match text.find('<') {
None => {
res.push_str(text);
break;
}
(ranges, text)
Some(i) => {
res.push_str(&text[..i]);
text = &text[i..];
if text.starts_with(&open) {
text = &text[open.len()..];
let from = TextUnit::of_str(&res);
stack.push(from);
} else if text.starts_with(&close) {
text = &text[close.len()..];
let from = stack
.pop()
.unwrap_or_else(|| panic!("unmatched </{}>", tag));
let to = TextUnit::of_str(&res);
ranges.push(TextRange::from_to(from, to));
}
}
}
}
assert!(stack.is_empty(), "unmatched <{}>", tag);
(ranges, res)
}
pub fn add_cursor(text: &str, offset: TextUnit) -> String {