mirror of
https://github.com/helix-editor/helix.git
synced 2025-04-06 12:27:42 +03:00
Handle snippets for LSPs not providing offsets for completion
This commit is contained in:
parent
0d924255e4
commit
8c2e447b16
2 changed files with 60 additions and 66 deletions
|
@ -252,26 +252,17 @@ pub mod util {
|
||||||
pub fn generate_transaction_from_completion_edit(
|
pub fn generate_transaction_from_completion_edit(
|
||||||
doc: &Rope,
|
doc: &Rope,
|
||||||
selection: &Selection,
|
selection: &Selection,
|
||||||
edit: lsp::TextEdit,
|
start_offset: i128,
|
||||||
offset_encoding: OffsetEncoding,
|
end_offset: i128,
|
||||||
|
new_text: String,
|
||||||
) -> Transaction {
|
) -> Transaction {
|
||||||
let replacement: Option<Tendril> = if edit.new_text.is_empty() {
|
let replacement: Option<Tendril> = if new_text.is_empty() {
|
||||||
None
|
None
|
||||||
} else {
|
} else {
|
||||||
Some(edit.new_text.into())
|
Some(new_text.into())
|
||||||
};
|
};
|
||||||
|
|
||||||
let text = doc.slice(..);
|
let text = doc.slice(..);
|
||||||
let primary_cursor = selection.primary().cursor(text);
|
|
||||||
|
|
||||||
let start_offset = match lsp_pos_to_pos(doc, edit.range.start, offset_encoding) {
|
|
||||||
Some(start) => start as i128 - primary_cursor as i128,
|
|
||||||
None => return Transaction::new(doc),
|
|
||||||
};
|
|
||||||
let end_offset = match lsp_pos_to_pos(doc, edit.range.end, offset_encoding) {
|
|
||||||
Some(end) => end as i128 - primary_cursor as i128,
|
|
||||||
None => return Transaction::new(doc),
|
|
||||||
};
|
|
||||||
|
|
||||||
Transaction::change_by_selection(doc, selection, |range| {
|
Transaction::change_by_selection(doc, selection, |range| {
|
||||||
let cursor = range.cursor(text);
|
let cursor = range.cursor(text);
|
||||||
|
@ -288,23 +279,13 @@ pub mod util {
|
||||||
pub fn generate_transaction_from_snippet(
|
pub fn generate_transaction_from_snippet(
|
||||||
doc: &Rope,
|
doc: &Rope,
|
||||||
selection: &Selection,
|
selection: &Selection,
|
||||||
edit_range: &lsp::Range,
|
start_offset: i128,
|
||||||
|
end_offset: i128,
|
||||||
snippet: snippet::Snippet,
|
snippet: snippet::Snippet,
|
||||||
line_ending: &str,
|
line_ending: &str,
|
||||||
include_placeholder: bool,
|
include_placeholder: bool,
|
||||||
offset_encoding: OffsetEncoding,
|
|
||||||
) -> Transaction {
|
) -> Transaction {
|
||||||
let text = doc.slice(..);
|
let text = doc.slice(..);
|
||||||
let primary_cursor = selection.primary().cursor(text);
|
|
||||||
|
|
||||||
let start_offset = match lsp_pos_to_pos(doc, edit_range.start, offset_encoding) {
|
|
||||||
Some(start) => start as i128 - primary_cursor as i128,
|
|
||||||
None => return Transaction::new(doc),
|
|
||||||
};
|
|
||||||
let end_offset = match lsp_pos_to_pos(doc, edit_range.end, offset_encoding) {
|
|
||||||
Some(end) => end as i128 - primary_cursor as i128,
|
|
||||||
None => return Transaction::new(doc),
|
|
||||||
};
|
|
||||||
|
|
||||||
// For each cursor store offsets for the first tabstop
|
// For each cursor store offsets for the first tabstop
|
||||||
let mut cursor_tabstop_offsets = Vec::<SmallVec<[(i128, i128); 1]>>::new();
|
let mut cursor_tabstop_offsets = Vec::<SmallVec<[(i128, i128); 1]>>::new();
|
||||||
|
|
|
@ -121,8 +121,9 @@ impl Completion {
|
||||||
include_placeholder: bool,
|
include_placeholder: bool,
|
||||||
) -> Transaction {
|
) -> Transaction {
|
||||||
use helix_lsp::snippet;
|
use helix_lsp::snippet;
|
||||||
|
let selection = doc.selection(view_id);
|
||||||
|
|
||||||
if let Some(edit) = &item.text_edit {
|
let (start_offset, end_offset, new_text) = if let Some(edit) = &item.text_edit {
|
||||||
let edit = match edit {
|
let edit = match edit {
|
||||||
lsp::CompletionTextEdit::Edit(edit) => edit.clone(),
|
lsp::CompletionTextEdit::Edit(edit) => edit.clone(),
|
||||||
lsp::CompletionTextEdit::InsertAndReplace(item) => {
|
lsp::CompletionTextEdit::InsertAndReplace(item) => {
|
||||||
|
@ -130,46 +131,27 @@ impl Completion {
|
||||||
lsp::TextEdit::new(item.replace, item.new_text.clone())
|
lsp::TextEdit::new(item.replace, item.new_text.clone())
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
let text = doc.text().slice(..);
|
||||||
|
let primary_cursor = selection.primary().cursor(text);
|
||||||
|
|
||||||
if matches!(item.kind, Some(lsp::CompletionItemKind::SNIPPET))
|
let start_offset =
|
||||||
|| matches!(
|
match util::lsp_pos_to_pos(doc.text(), edit.range.start, offset_encoding) {
|
||||||
item.insert_text_format,
|
Some(start) => start as i128 - primary_cursor as i128,
|
||||||
Some(lsp::InsertTextFormat::SNIPPET)
|
None => return Transaction::new(doc.text()),
|
||||||
)
|
};
|
||||||
{
|
let end_offset =
|
||||||
match snippet::parse(&edit.new_text) {
|
match util::lsp_pos_to_pos(doc.text(), edit.range.end, offset_encoding) {
|
||||||
Ok(snippet) => util::generate_transaction_from_snippet(
|
Some(end) => end as i128 - primary_cursor as i128,
|
||||||
doc.text(),
|
None => return Transaction::new(doc.text()),
|
||||||
doc.selection(view_id),
|
};
|
||||||
&edit.range,
|
|
||||||
snippet,
|
(start_offset, end_offset, edit.new_text)
|
||||||
doc.line_ending.as_str(),
|
|
||||||
include_placeholder,
|
|
||||||
offset_encoding,
|
|
||||||
),
|
|
||||||
Err(err) => {
|
|
||||||
log::error!(
|
|
||||||
"Failed to parse snippet: {:?}, remaining output: {}",
|
|
||||||
&edit.new_text,
|
|
||||||
err
|
|
||||||
);
|
|
||||||
Transaction::new(doc.text())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
util::generate_transaction_from_completion_edit(
|
|
||||||
doc.text(),
|
|
||||||
doc.selection(view_id),
|
|
||||||
edit,
|
|
||||||
offset_encoding, // TODO: should probably transcode in Client
|
|
||||||
)
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
let text = item.insert_text.as_ref().unwrap_or(&item.label);
|
let new_text = item.insert_text.as_ref().unwrap_or(&item.label);
|
||||||
// Some LSPs just give you an insertText with no offset ¯\_(ツ)_/¯
|
// Some LSPs just give you an insertText with no offset ¯\_(ツ)_/¯
|
||||||
// in these cases we need to check for a common prefix and remove it
|
// in these cases we need to check for a common prefix and remove it
|
||||||
let prefix = Cow::from(doc.text().slice(start_offset..trigger_offset));
|
let prefix = Cow::from(doc.text().slice(start_offset..trigger_offset));
|
||||||
let text = text.trim_start_matches::<&str>(&prefix);
|
let new_text = new_text.trim_start_matches::<&str>(&prefix);
|
||||||
|
|
||||||
// TODO: this needs to be true for the numbers to work out correctly
|
// TODO: this needs to be true for the numbers to work out correctly
|
||||||
// in the closure below. It's passed in to a callback as this same
|
// in the closure below. It's passed in to a callback as this same
|
||||||
|
@ -182,11 +164,42 @@ impl Completion {
|
||||||
== trigger_offset
|
== trigger_offset
|
||||||
);
|
);
|
||||||
|
|
||||||
Transaction::change_by_selection(doc.text(), doc.selection(view_id), |range| {
|
(0, 0, new_text.into())
|
||||||
let cursor = range.cursor(doc.text().slice(..));
|
};
|
||||||
|
|
||||||
(cursor, cursor, Some(text.into()))
|
if matches!(item.kind, Some(lsp::CompletionItemKind::SNIPPET))
|
||||||
})
|
|| matches!(
|
||||||
|
item.insert_text_format,
|
||||||
|
Some(lsp::InsertTextFormat::SNIPPET)
|
||||||
|
)
|
||||||
|
{
|
||||||
|
match snippet::parse(&new_text) {
|
||||||
|
Ok(snippet) => util::generate_transaction_from_snippet(
|
||||||
|
doc.text(),
|
||||||
|
selection,
|
||||||
|
start_offset,
|
||||||
|
end_offset,
|
||||||
|
snippet,
|
||||||
|
doc.line_ending.as_str(),
|
||||||
|
include_placeholder,
|
||||||
|
),
|
||||||
|
Err(err) => {
|
||||||
|
log::error!(
|
||||||
|
"Failed to parse snippet: {:?}, remaining output: {}",
|
||||||
|
&new_text,
|
||||||
|
err
|
||||||
|
);
|
||||||
|
Transaction::new(doc.text())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
util::generate_transaction_from_completion_edit(
|
||||||
|
doc.text(),
|
||||||
|
selection,
|
||||||
|
start_offset,
|
||||||
|
end_offset,
|
||||||
|
new_text,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue