aboutsummaryrefslogtreecommitdiff
path: root/crates/ra_lsp_server/src/conv.rs
diff options
context:
space:
mode:
Diffstat (limited to 'crates/ra_lsp_server/src/conv.rs')
-rw-r--r--crates/ra_lsp_server/src/conv.rs106
1 files changed, 92 insertions, 14 deletions
diff --git a/crates/ra_lsp_server/src/conv.rs b/crates/ra_lsp_server/src/conv.rs
index 1318a1738..ee503633d 100644
--- a/crates/ra_lsp_server/src/conv.rs
+++ b/crates/ra_lsp_server/src/conv.rs
@@ -227,22 +227,57 @@ impl ConvWith<(&LineIndex, LineEndings)> for &AtomTextEdit {
227 } 227 }
228} 228}
229 229
230impl ConvWith<&LineIndex> for Fold { 230pub(crate) struct FoldConvCtx<'a> {
231 pub(crate) text: &'a str,
232 pub(crate) line_index: &'a LineIndex,
233 pub(crate) line_folding_only: bool,
234}
235
236impl ConvWith<&FoldConvCtx<'_>> for Fold {
231 type Output = lsp_types::FoldingRange; 237 type Output = lsp_types::FoldingRange;
232 238
233 fn conv_with(self, line_index: &LineIndex) -> lsp_types::FoldingRange { 239 fn conv_with(self, ctx: &FoldConvCtx) -> lsp_types::FoldingRange {
234 let range = self.range.conv_with(&line_index); 240 let kind = match self.kind {
235 lsp_types::FoldingRange { 241 FoldKind::Comment => Some(lsp_types::FoldingRangeKind::Comment),
236 start_line: range.start.line, 242 FoldKind::Imports => Some(lsp_types::FoldingRangeKind::Imports),
237 start_character: Some(range.start.character), 243 FoldKind::Mods => None,
238 end_line: range.end.line, 244 FoldKind::Block => None,
239 end_character: Some(range.end.character), 245 };
240 kind: match self.kind { 246
241 FoldKind::Comment => Some(lsp_types::FoldingRangeKind::Comment), 247 let range = self.range.conv_with(&ctx.line_index);
242 FoldKind::Imports => Some(lsp_types::FoldingRangeKind::Imports), 248
243 FoldKind::Mods => None, 249 if ctx.line_folding_only {
244 FoldKind::Block => None, 250 // Clients with line_folding_only == true (such as VSCode) will fold the whole end line
245 }, 251 // even if it contains text not in the folding range. To prevent that we exclude
252 // range.end.line from the folding region if there is more text after range.end
253 // on the same line.
254 let has_more_text_on_end_line = ctx.text
255 [TextRange::from_to(self.range.end(), TextUnit::of_str(ctx.text))]
256 .chars()
257 .take_while(|it| *it != '\n')
258 .any(|it| !it.is_whitespace());
259
260 let end_line = if has_more_text_on_end_line {
261 range.end.line.saturating_sub(1)
262 } else {
263 range.end.line
264 };
265
266 lsp_types::FoldingRange {
267 start_line: range.start.line,
268 start_character: None,
269 end_line,
270 end_character: None,
271 kind,
272 }
273 } else {
274 lsp_types::FoldingRange {
275 start_line: range.start.line,
276 start_character: Some(range.start.character),
277 end_line: range.end.line,
278 end_character: Some(range.end.character),
279 kind,
280 }
246 } 281 }
247 } 282 }
248} 283}
@@ -512,3 +547,46 @@ where
512 self.map(|it| it.try_conv_with(ctx)).collect() 547 self.map(|it| it.try_conv_with(ctx)).collect()
513 } 548 }
514} 549}
550
551#[cfg(test)]
552mod tests {
553 use super::*;
554 use test_utils::extract_ranges;
555
556 #[test]
557 fn conv_fold_line_folding_only_fixup() {
558 let text = r#"<fold>mod a;
559mod b;
560mod c;</fold>
561
562fn main() <fold>{
563 if cond <fold>{
564 a::do_a();
565 }</fold> else <fold>{
566 b::do_b();
567 }</fold>
568}</fold>"#;
569
570 let (ranges, text) = extract_ranges(text, "fold");
571 assert_eq!(ranges.len(), 4);
572 let folds = vec![
573 Fold { range: ranges[0], kind: FoldKind::Mods },
574 Fold { range: ranges[1], kind: FoldKind::Block },
575 Fold { range: ranges[2], kind: FoldKind::Block },
576 Fold { range: ranges[3], kind: FoldKind::Block },
577 ];
578
579 let line_index = LineIndex::new(&text);
580 let ctx = FoldConvCtx { text: &text, line_index: &line_index, line_folding_only: true };
581 let converted: Vec<_> = folds.into_iter().map_conv_with(&ctx).collect();
582
583 let expected_lines = [(0, 2), (4, 10), (5, 6), (7, 9)];
584 assert_eq!(converted.len(), expected_lines.len());
585 for (folding_range, (start_line, end_line)) in converted.iter().zip(expected_lines.iter()) {
586 assert_eq!(folding_range.start_line, *start_line);
587 assert_eq!(folding_range.start_character, None);
588 assert_eq!(folding_range.end_line, *end_line);
589 assert_eq!(folding_range.end_character, None);
590 }
591 }
592}