Improve relative indent computation.
Add tests to ensure that relative & absolute indent computation are consistent.
This commit is contained in:
parent
d29a66f267
commit
559bfc1f5e
2 changed files with 127 additions and 51 deletions
|
@ -210,6 +210,46 @@ fn whitespace_with_same_width(text: RopeSlice) -> String {
|
|||
s
|
||||
}
|
||||
|
||||
fn add_indent_level(
|
||||
mut base_indent: String,
|
||||
added_indent_level: isize,
|
||||
indent_style: &IndentStyle,
|
||||
tab_width: usize,
|
||||
) -> String {
|
||||
if added_indent_level >= 0 {
|
||||
// Adding a non-negative indent is easy, we can simply append the indent string
|
||||
base_indent.push_str(&indent_style.as_str().repeat(added_indent_level as usize));
|
||||
base_indent
|
||||
} else {
|
||||
// In this case, we want to return a prefix of `base_indent`.
|
||||
// Since the width of a tab depends on its offset, we cannot simply iterate over
|
||||
// the chars of `base_indent` in reverse until we have the desired indent reduction,
|
||||
// instead we iterate over them twice in forward direction.
|
||||
let mut base_indent_width: usize = 0;
|
||||
for c in base_indent.chars() {
|
||||
base_indent_width += match c {
|
||||
'\t' => tab_width_at(base_indent_width, tab_width as u16),
|
||||
// This is only true since `base_indent` consists only of tabs and spaces
|
||||
_ => 1,
|
||||
};
|
||||
}
|
||||
let target_indent_width = base_indent_width
|
||||
.saturating_sub((-added_indent_level) as usize * indent_style.indent_width(tab_width));
|
||||
let mut reduced_indent_width = 0;
|
||||
for (i, c) in base_indent.char_indices() {
|
||||
if reduced_indent_width >= target_indent_width {
|
||||
base_indent.truncate(i);
|
||||
return base_indent;
|
||||
}
|
||||
reduced_indent_width += match c {
|
||||
'\t' => tab_width_at(base_indent_width, tab_width as u16),
|
||||
_ => 1,
|
||||
};
|
||||
}
|
||||
base_indent
|
||||
}
|
||||
}
|
||||
|
||||
/// Computes for node and all ancestors whether they are the first node on their line.
|
||||
/// The first entry in the return value represents the root node, the last one the node itself
|
||||
fn get_first_in_line(mut node: Node, new_line_byte_pos: Option<usize>) -> Vec<bool> {
|
||||
|
@ -334,59 +374,25 @@ impl<'a> Indentation<'a> {
|
|||
// we can simply take `other_leading_whitespace` and add some indent / outdent to it (in the second
|
||||
// case, the alignment should already be accounted for in `other_leading_whitespace`).
|
||||
let indent_diff = self.net_indent() - other_computed_indent.net_indent();
|
||||
if indent_diff >= 0 {
|
||||
let mut indent = other_leading_whitespace.to_string();
|
||||
indent.push_str(&indent_style.as_str().repeat(indent_diff as usize));
|
||||
Some(indent)
|
||||
} else {
|
||||
// It's not entirely clear how to subtract a given indent level from the other line if its indentation is
|
||||
// complex (e.g. a weird alignment or a mixture of tabs and spaces). Therefore, we only consider the indent level
|
||||
// of `baseline_leading_whitespace`, add `indent_diff` to it and convert this indent level back to a string. If we ever encounter
|
||||
// cases where some other behavior is expected, the behavior for strings that don't exactly correspond to some indent
|
||||
// level could be re-evaluated.
|
||||
let actual_baseline_indent_level = indent_level_for_line(
|
||||
other_leading_whitespace,
|
||||
tab_width,
|
||||
indent_style.indent_width(tab_width),
|
||||
);
|
||||
let total_indent = actual_baseline_indent_level as isize + indent_diff;
|
||||
if total_indent < 0 {
|
||||
log::warn!(
|
||||
"Computed negative indent during a relative indent computation (actual baseline indent: {}, computed baseline indent: {}, computed line indent: {})",
|
||||
actual_baseline_indent_level,
|
||||
other_computed_indent.net_indent(),
|
||||
self.net_indent(),
|
||||
);
|
||||
Some(String::new())
|
||||
} else {
|
||||
Some(indent_style.as_str().repeat(total_indent as usize))
|
||||
}
|
||||
}
|
||||
} else if self.align.is_some() {
|
||||
Some(self.to_string(indent_style))
|
||||
Some(add_indent_level(
|
||||
String::from(other_leading_whitespace),
|
||||
indent_diff,
|
||||
indent_style,
|
||||
tab_width,
|
||||
))
|
||||
} else {
|
||||
// If the other line has some alignment and `self` does not, there is no reasonable way to take
|
||||
// into account `other_leading_whitespace`.
|
||||
// If the alignment of both lines is different, we cannot compare their indentation in any meaningful way
|
||||
None
|
||||
}
|
||||
}
|
||||
pub fn to_string(&self, indent_style: &IndentStyle) -> String {
|
||||
let indent = self.indent_always + self.indent;
|
||||
let outdent = self.outdent_always + self.outdent;
|
||||
|
||||
let indent_level = if indent >= outdent {
|
||||
indent - outdent
|
||||
} else {
|
||||
log::warn!("Encountered more outdent than indent nodes while calculating indentation: {} outdent, {} indent", self.outdent, self.indent);
|
||||
0
|
||||
};
|
||||
let mut indent_string = if let Some(align) = self.align {
|
||||
whitespace_with_same_width(align)
|
||||
} else {
|
||||
String::new()
|
||||
};
|
||||
indent_string.push_str(&indent_style.as_str().repeat(indent_level));
|
||||
indent_string
|
||||
pub fn to_string(&self, indent_style: &IndentStyle, tab_width: usize) -> String {
|
||||
add_indent_level(
|
||||
self.align
|
||||
.map_or_else(String::new, whitespace_with_same_width),
|
||||
self.net_indent(),
|
||||
indent_style,
|
||||
tab_width,
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -992,7 +998,7 @@ pub fn indent_for_newline(
|
|||
break;
|
||||
}
|
||||
}
|
||||
return indent.to_string(indent_style);
|
||||
return indent.to_string(indent_style, tab_width);
|
||||
};
|
||||
}
|
||||
// Fallback in case we either don't have indent queries or they failed for some reason
|
||||
|
@ -1188,4 +1194,74 @@ mod test {
|
|||
)
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_relative_indent() {
|
||||
let indent_style = IndentStyle::Spaces(4);
|
||||
let tab_width: usize = 4;
|
||||
let no_align = [
|
||||
Indentation::default(),
|
||||
Indentation {
|
||||
indent: 1,
|
||||
..Default::default()
|
||||
},
|
||||
Indentation {
|
||||
indent: 5,
|
||||
outdent: 1,
|
||||
..Default::default()
|
||||
},
|
||||
];
|
||||
let align = no_align.clone().map(|indent| Indentation {
|
||||
align: Some(RopeSlice::from("12345")),
|
||||
..indent
|
||||
});
|
||||
let different_align = Indentation {
|
||||
align: Some(RopeSlice::from("123456")),
|
||||
..Default::default()
|
||||
};
|
||||
|
||||
// Check that relative and absolute indentation computation are the same when the line we compare to is
|
||||
// indented as we expect.
|
||||
let check_consistency = |indent: &Indentation, other: &Indentation| {
|
||||
assert_eq!(
|
||||
indent.relative_indent(
|
||||
other,
|
||||
RopeSlice::from(other.to_string(&indent_style, tab_width).as_str()),
|
||||
&indent_style,
|
||||
tab_width
|
||||
),
|
||||
Some(indent.to_string(&indent_style, tab_width))
|
||||
);
|
||||
};
|
||||
for a in &no_align {
|
||||
for b in &no_align {
|
||||
check_consistency(a, b);
|
||||
}
|
||||
}
|
||||
for a in &align {
|
||||
for b in &align {
|
||||
check_consistency(a, b);
|
||||
}
|
||||
}
|
||||
|
||||
// Relative indent computation makes no sense if the alignment differs
|
||||
assert_eq!(
|
||||
align[0].relative_indent(
|
||||
&no_align[0],
|
||||
RopeSlice::from(" "),
|
||||
&indent_style,
|
||||
tab_width
|
||||
),
|
||||
None
|
||||
);
|
||||
assert_eq!(
|
||||
align[0].relative_indent(
|
||||
&different_align,
|
||||
RopeSlice::from(" "),
|
||||
&indent_style,
|
||||
tab_width
|
||||
),
|
||||
None
|
||||
);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -218,7 +218,7 @@ fn test_treesitter_indent(
|
|||
false,
|
||||
)
|
||||
.unwrap()
|
||||
.to_string(&indent_style);
|
||||
.to_string(&indent_style, tab_width);
|
||||
assert!(
|
||||
line.get_slice(..pos).map_or(false, |s| s == suggested_indent),
|
||||
"Wrong indentation for file {:?} on line {}:\n\"{}\" (original line)\n\"{}\" (suggested indentation)\n",
|
||||
|
|
Loading…
Reference in a new issue