use deno_core::error::AnyError;
use imara_diff::Algorithm;
use imara_diff::Diff;
use imara_diff::InternedInput;
use text_size::TextRange;
use text_size::TextSize;
use tower_lsp::jsonrpc;
use tower_lsp::lsp_types as lsp;
use tower_lsp::lsp_types::TextEdit;
use crate::util::text_encoding::Utf16Map;
#[derive(Debug, Clone, Default, Eq, PartialEq)]
pub struct LineIndex {
inner: Utf16Map,
}
impl LineIndex {
pub fn new(text: &str) -> LineIndex {
LineIndex {
inner: Utf16Map::new(text),
}
}
pub fn get_text_range(
&self,
range: lsp::Range,
) -> Result<TextRange, AnyError> {
let start = self.offset(range.start)?;
let end = self.offset(range.end)?;
Ok(TextRange::new(start, end))
}
pub fn offset(&self, position: lsp::Position) -> Result<TextSize, AnyError> {
self.inner.offset(position.line, position.character)
}
pub fn offset_tsc(&self, position: lsp::Position) -> jsonrpc::Result<u32> {
self
.inner
.offset_utf16(position.line, position.character)
.map(|ts| ts.into())
.map_err(|err| jsonrpc::Error::invalid_params(err.to_string()))
}
pub fn position_utf16(&self, offset: TextSize) -> lsp::Position {
let lc = self.inner.position_utf16(offset);
lsp::Position {
line: lc.line_index as u32,
character: lc.column_index as u32,
}
}
pub fn line_length_utf16(&self, line: u32) -> TextSize {
self.inner.line_length_utf16(line)
}
pub fn text_content_length_utf16(&self) -> TextSize {
self.inner.text_content_length_utf16()
}
}
pub fn get_edits(a: &str, b: &str, line_index: &LineIndex) -> Vec<TextEdit> {
if a == b {
return vec![];
}
let b_lines = b.chars().filter(|c| *c == '\n').count();
if b_lines > 10000 || b_lines > line_index.inner.utf8_offsets_len() * 3 {
return vec![TextEdit {
range: lsp::Range {
start: lsp::Position::new(0, 0),
end: line_index.position_utf16(TextSize::from(a.len() as u32)),
},
new_text: b.to_string(),
}];
}
let input = InternedInput::new(a, b);
let mut diff = Diff::compute(Algorithm::Histogram, &input);
diff.postprocess_lines(&input);
let a_line_offsets: Vec<u32> = {
let mut offsets = vec![0u32];
let mut pos = 0u32;
for line_token in input.before.iter() {
let line_str = input.interner[*line_token];
pos += line_str.encode_utf16().count() as u32;
offsets.push(pos);
}
offsets
};
let mut text_edits = Vec::<TextEdit>::new();
for hunk in diff.hunks() {
let del_start_line = hunk.before.start as usize;
let del_end_line = hunk.before.end as usize;
let start_offset = TextSize::from(a_line_offsets[del_start_line]);
let end_offset = TextSize::from(a_line_offsets[del_end_line]);
let start = line_index.position_utf16(start_offset);
let end = line_index.position_utf16(end_offset);
let range = lsp::Range { start, end };
let mut new_text = String::new();
for ins_idx in hunk.after.start..hunk.after.end {
new_text.push_str(input.interner[input.after[ins_idx as usize]]);
}
text_edits.push(TextEdit { range, new_text });
}
text_edits
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_get_edits() {
let a = "abcdefg";
let b = "a\nb\nchije\nfg\n";
let actual = get_edits(a, b, &LineIndex::new(a));
assert_eq!(
actual,
vec![TextEdit {
range: lsp::Range {
start: lsp::Position {
line: 0,
character: 0
},
end: lsp::Position {
line: 0,
character: 7
}
},
new_text: "a\nb\nchije\nfg\n".to_string()
}]
);
}
#[test]
fn test_get_edits_mbc() {
let a = "const bar = \"ππΊπΈπ\";\nconsole.log('hello deno')\n";
let b = "const bar = \"ππΊπΈπ\";\nconsole.log(\"hello deno\");\n";
let actual = get_edits(a, b, &LineIndex::new(a));
assert_eq!(
actual,
vec![TextEdit {
range: lsp::Range {
start: lsp::Position {
line: 1,
character: 0
},
end: lsp::Position {
line: 2,
character: 0
}
},
new_text: "console.log(\"hello deno\");\n".to_string()
}]
)
}
#[test]
fn test_get_edits_no_changes() {
let a = "hello world\n";
let actual = get_edits(a, a, &LineIndex::new(a));
assert_eq!(actual, vec![]);
}
#[test]
fn test_get_edits_insert_line() {
let a = "line1\nline3\n";
let b = "line1\nline2\nline3\n";
let actual = get_edits(a, b, &LineIndex::new(a));
assert_eq!(
actual,
vec![TextEdit {
range: lsp::Range {
start: lsp::Position {
line: 1,
character: 0
},
end: lsp::Position {
line: 1,
character: 0
}
},
new_text: "line2\n".to_string()
}]
);
}
}