Skip to content

Commit

Permalink
lsp: Improve line ending handling when generating TextEdit
Browse files Browse the repository at this point in the history
  • Loading branch information
archseer committed Sep 6, 2021
1 parent 08967ba commit 57ed518
Show file tree
Hide file tree
Showing 2 changed files with 41 additions and 3 deletions.
10 changes: 7 additions & 3 deletions helix-lsp/src/client.rs
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ use crate::{
Call, Error, OffsetEncoding, Result,
};

use helix_core::{chars::char_is_line_ending, find_root, ChangeSet, Rope};
use helix_core::{find_root, ChangeSet, Rope};
use jsonrpc_core as jsonrpc;
use lsp_types as lsp;
use serde_json::Value;
Expand Down Expand Up @@ -356,7 +356,6 @@ impl Client {
//
// Calculation is therefore a bunch trickier.

// TODO: stolen from syntax.rs, share
use helix_core::RopeSlice;
fn traverse(pos: lsp::Position, text: RopeSlice) -> lsp::Position {
let lsp::Position {
Expand All @@ -366,7 +365,12 @@ impl Client {

let mut chars = text.chars().peekable();
while let Some(ch) = chars.next() {
if char_is_line_ending(ch) && !(ch == '\r' && chars.peek() == Some(&'\n')) {
// LSP only considers \n, \r or \r\n as line endings
if ch == '\n' || ch == '\r' {
// consume a \r\n
if chars.peek() == Some(&'\n') {
chars.next();
}
line += 1;
character = 0;
} else {
Expand Down
34 changes: 34 additions & 0 deletions helix-view/src/document.rs
Original file line number Diff line number Diff line change
Expand Up @@ -891,6 +891,40 @@ impl Default for Document {
mod test {
use super::*;

#[test]
fn changeset_to_changes_ignore_line_endings() {
use helix_lsp::{lsp, Client, OffsetEncoding};
let text = Rope::from("hello\r\nworld");
let mut doc = Document::from(text, None);
let view = ViewId::default();
doc.set_selection(view, Selection::single(0, 0));

let transaction =
Transaction::change(doc.text(), vec![(5, 7, Some("\n".into()))].into_iter());
let old_doc = doc.text().clone();
doc.apply(&transaction, view);
let changes = Client::changeset_to_changes(
&old_doc,
doc.text(),
transaction.changes(),
OffsetEncoding::Utf8,
);

assert_eq!(doc.text(), "hello\nworld");

assert_eq!(
changes,
&[lsp::TextDocumentContentChangeEvent {
range: Some(lsp::Range::new(
lsp::Position::new(0, 5),
lsp::Position::new(1, 0)
)),
text: "\n".into(),
range_length: None,
}]
);
}

#[test]
fn changeset_to_changes() {
use helix_lsp::{lsp, Client, OffsetEncoding};
Expand Down

0 comments on commit 57ed518

Please sign in to comment.