From b11b930d9ab4469451d5b6ce757dac2a0f15cc9d Mon Sep 17 00:00:00 2001 From: Guldoman Date: Sat, 24 Feb 2024 05:37:34 +0100 Subject: [PATCH] Apply format `TextEdit`s in reverse order The `TextEdit` ranges don't consider the intermediate doc states, so applying them in reverse order (from bottom to top) should result in edits occurring at the intended locations. --- init.lua | 18 +++++++++++++----- 1 file changed, 13 insertions(+), 5 deletions(-) diff --git a/init.lua b/init.lua index 68bf8ca..458e2d7 100644 --- a/init.lua +++ b/init.lua @@ -386,8 +386,9 @@ end ---@param doc core.doc ---@param text_edit table ---@param is_snippet boolean +---@param update_cursor_position boolean ---@return boolean True on success -local function apply_edit(server, doc, text_edit, is_snippet) +local function apply_edit(server, doc, text_edit, is_snippet, update_cursor_position) local range = nil if text_edit.range then @@ -433,7 +434,9 @@ local function apply_edit(server, doc, text_edit, is_snippet) end doc:insert(line1, col1, text) - doc:set_selection(line2, col1+#text, line2, col1+#text) + if update_cursor_position then + doc:set_selection(line2, col1+#text, line2, col1+#text) + end return true end @@ -513,7 +516,7 @@ local function autocomplete_onselect(index, item) if dv then local is_snippet = completion.insertTextFormat and completion.insertTextFormat == Server.insert_text_format.Snippet - local edit_applied = apply_edit(item.data.server, dv.doc, completion.textEdit, is_snippet) + local edit_applied = apply_edit(item.data.server, dv.doc, completion.textEdit, is_snippet, true) if edit_applied then -- Retrigger code completion if last char is a trigger -- this is useful for example with clangd when autocompleting @@ -1947,8 +1950,13 @@ function lsp.request_document_format(doc) if response.error and response.error.message then log(server, "Error formatting: " .. response.error.message) elseif response.result and #response.result > 0 then - for _, result in pairs(response.result) do - apply_edit(server, doc, result) + -- Apply edits in reverse, as the ranges don't consider + -- the intermediate states. + -- Consider the TextEdits as already sorted. + -- If there are servers that don't sort their TextEdits, + -- we'll add sorting code. + for i=#response.result,1,-1 do + apply_edit(server, doc, response.result[i], false, false) end log(server, "Formatted document") else