Auto merge of #18243 - krobelus:clamp-position-character, r=Veykril

Clamp Position::character to line length

LSP says about Position::character

> If the character value is greater than the line length it defaults back to the line length.

but from_proto::offset() doesn't implement this.

A client might for example request code actions for a whole line by sending
Position::character=99999.  I don't think there is ever a reason (besides laziness) why the
client can't specify the line length instead but I guess we should not crash but follow protocol.

Not sure how to update Cargo.lock (lib/README.md doesn't say how).

Fixes #18240
This commit is contained in:
bors 2024-10-18 11:23:05 +00:00
commit 3bda0cc2c0
3 changed files with 32 additions and 1 deletions

View file

@ -1,6 +1,6 @@
[package] [package]
name = "line-index" name = "line-index"
version = "0.1.1" version = "0.1.2"
description = "Maps flat `TextSize` offsets to/from `(line, column)` representation." description = "Maps flat `TextSize` offsets to/from `(line, column)` representation."
license = "MIT OR Apache-2.0" license = "MIT OR Apache-2.0"
repository = "https://github.com/rust-lang/rust-analyzer/tree/master/lib/line-index" repository = "https://github.com/rust-lang/rust-analyzer/tree/master/lib/line-index"

View file

@ -177,6 +177,14 @@ impl LineIndex {
Some(LineCol { line: line_col.line, col }) Some(LineCol { line: line_col.line, col })
} }
/// Returns the given line's range.
pub fn line(&self, line: u32) -> Option<TextRange> {
let start = self.start_offset(line as usize)?;
let next_newline = self.newlines.get(line as usize).copied().unwrap_or(self.len);
let line_length = next_newline - start;
Some(TextRange::new(start, start + line_length))
}
/// Given a range [start, end), returns a sorted iterator of non-empty ranges [start, x1), [x1, /// Given a range [start, end), returns a sorted iterator of non-empty ranges [start, x1), [x1,
/// x2), ..., [xn, end) where all the xi, which are positions of newlines, are inside the range /// x2), ..., [xn, end) where all the xi, which are positions of newlines, are inside the range
/// [start, end). /// [start, end).

View file

@ -195,3 +195,26 @@ fn test_every_chars() {
} }
} }
} }
#[test]
fn test_line() {
use text_size::TextRange;
macro_rules! validate {
($text:expr, $line:expr, $expected_start:literal .. $expected_end:literal) => {
let line_index = LineIndex::new($text);
assert_eq!(
line_index.line($line),
Some(TextRange::new(
TextSize::from($expected_start),
TextSize::from($expected_end)
))
);
};
}
validate!("", 0, 0..0);
validate!("\n", 1, 1..1);
validate!("\nabc", 1, 1..4);
validate!("\nabc\ndef", 1, 1..5);
}