mirror of
https://github.com/neovim/neovim.git
synced 2025-10-25 20:07:09 +00:00
fix(lsp): handle multiple clients with incremental sync (#19658)
The change tracking used a single lines/lines_tmp table to track changes to a buffer. If multiple clients using incremental sync are connected to a buffer, they both made changes to the same lines table. That resulted in an inconsistent state. This commit changes the didChange handling to group clients by synchronization scheme and offset encoding. This avoids computing the diff multiple times for clients using the same scheme and resolves the lines/lines_tmp conflicts. Fixes https://github.com/neovim/neovim/issues/19325
This commit is contained in:
committed by
GitHub
parent
3030b4d653
commit
bebfbfab3b
@@ -338,116 +338,84 @@ end
|
|||||||
|
|
||||||
local changetracking = {}
|
local changetracking = {}
|
||||||
do
|
do
|
||||||
--@private
|
---@private
|
||||||
--- client_id → state
|
|
||||||
---
|
---
|
||||||
--- state
|
--- LSP has 3 different sync modes:
|
||||||
--- use_incremental_sync: bool
|
--- - None (Servers will read the files themselves when needed)
|
||||||
--- buffers: bufnr -> buffer_state
|
--- - Full (Client sends the full buffer content on updates)
|
||||||
|
--- - Incremental (Client sends only the changed parts)
|
||||||
---
|
---
|
||||||
--- buffer_state
|
--- Changes are tracked per buffer.
|
||||||
--- pending_change?: function that the timer starts to trigger didChange
|
--- A buffer can have multiple clients attached and each client needs to send the changes
|
||||||
--- pending_changes: table (uri -> list of pending changeset tables));
|
--- To minimize the amount of changesets to compute, computation is grouped:
|
||||||
--- Only set if incremental_sync is used
|
|
||||||
---
|
---
|
||||||
--- timer?: uv_timer
|
--- None: One group for all clients
|
||||||
--- lines: table
|
--- Full: One group for all clients
|
||||||
local state_by_client = {}
|
--- Incremental: One group per `offset_encoding`
|
||||||
|
---
|
||||||
|
--- Sending changes can be debounced per buffer. To simplify the implementation the
|
||||||
|
--- smallest debounce interval is used and we don't group clients by different intervals.
|
||||||
|
---
|
||||||
|
--- @class CTGroup
|
||||||
|
--- @field sync_kind number TextDocumentSyncKind, considers config.flags.allow_incremental_sync
|
||||||
|
--- @field offset_encoding "utf-8"|"utf-16"|"utf-32"
|
||||||
|
---
|
||||||
|
--- @class CTBufferState
|
||||||
|
--- @field name string name of the buffer
|
||||||
|
--- @field lines string[] snapshot of buffer lines from last didChange
|
||||||
|
--- @field lines_tmp string[]
|
||||||
|
--- @field pending_changes table[] List of debounced changes in incremental sync mode
|
||||||
|
--- @field timer nil|userdata uv_timer
|
||||||
|
--- @field last_flush nil|number uv.hrtime of the last flush/didChange-notification
|
||||||
|
--- @field needs_flush boolean true if buffer updates haven't been sent to clients/servers yet
|
||||||
|
--- @field refs number how many clients are using this group
|
||||||
|
---
|
||||||
|
--- @class CTGroupState
|
||||||
|
--- @field buffers table<number, CTBufferState>
|
||||||
|
--- @field debounce number debounce duration in ms
|
||||||
|
--- @field clients table<number, table> clients using this state. {client_id, client}
|
||||||
|
|
||||||
---@private
|
---@private
|
||||||
function changetracking.init(client, bufnr)
|
---@param group CTGroup
|
||||||
local use_incremental_sync = (
|
---@return string
|
||||||
if_nil(client.config.flags.allow_incremental_sync, true)
|
local function group_key(group)
|
||||||
and vim.tbl_get(client.server_capabilities, 'textDocumentSync', 'change')
|
if group.sync_kind == protocol.TextDocumentSyncKind.Incremental then
|
||||||
== protocol.TextDocumentSyncKind.Incremental
|
return tostring(group.sync_kind) .. '\0' .. group.offset_encoding
|
||||||
)
|
end
|
||||||
local state = state_by_client[client.id]
|
return tostring(group.sync_kind)
|
||||||
if not state then
|
end
|
||||||
state = {
|
|
||||||
buffers = {},
|
---@private
|
||||||
debounce = client.config.flags.debounce_text_changes or 150,
|
---@type table<CTGroup, CTGroupState>
|
||||||
use_incremental_sync = use_incremental_sync,
|
local state_by_group = setmetatable({}, {
|
||||||
|
__index = function(tbl, k)
|
||||||
|
return rawget(tbl, group_key(k))
|
||||||
|
end,
|
||||||
|
__newindex = function(tbl, k, v)
|
||||||
|
rawset(tbl, group_key(k), v)
|
||||||
|
end,
|
||||||
|
})
|
||||||
|
|
||||||
|
---@private
|
||||||
|
---@return CTGroup
|
||||||
|
local function get_group(client)
|
||||||
|
local allow_inc_sync = if_nil(client.config.flags.allow_incremental_sync, true)
|
||||||
|
local change_capability = vim.tbl_get(client.server_capabilities, 'textDocumentSync', 'change')
|
||||||
|
local sync_kind = change_capability or protocol.TextDocumentSyncKind.None
|
||||||
|
if not allow_inc_sync and change_capability == protocol.TextDocumentSyncKind.Incremental then
|
||||||
|
sync_kind = protocol.TextDocumentSyncKind.Full
|
||||||
|
end
|
||||||
|
return {
|
||||||
|
sync_kind = sync_kind,
|
||||||
|
offset_encoding = client.offset_encoding,
|
||||||
}
|
}
|
||||||
state_by_client[client.id] = state
|
|
||||||
end
|
|
||||||
if not state.buffers[bufnr] then
|
|
||||||
local buf_state = {
|
|
||||||
name = api.nvim_buf_get_name(bufnr),
|
|
||||||
}
|
|
||||||
state.buffers[bufnr] = buf_state
|
|
||||||
if use_incremental_sync then
|
|
||||||
buf_state.lines = nvim_buf_get_lines(bufnr, 0, -1, true)
|
|
||||||
buf_state.lines_tmp = {}
|
|
||||||
buf_state.pending_changes = {}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
|
||||||
---@private
|
---@private
|
||||||
function changetracking._get_and_set_name(client, bufnr, name)
|
---@param state CTBufferState
|
||||||
local state = state_by_client[client.id] or {}
|
local function incremental_changes(state, encoding, bufnr, firstline, lastline, new_lastline)
|
||||||
local buf_state = (state.buffers or {})[bufnr]
|
local prev_lines = state.lines
|
||||||
local old_name = buf_state.name
|
local curr_lines = state.lines_tmp
|
||||||
buf_state.name = name
|
|
||||||
return old_name
|
|
||||||
end
|
|
||||||
|
|
||||||
---@private
|
|
||||||
function changetracking.reset_buf(client, bufnr)
|
|
||||||
changetracking.flush(client, bufnr)
|
|
||||||
local state = state_by_client[client.id]
|
|
||||||
if state and state.buffers then
|
|
||||||
local buf_state = state.buffers[bufnr]
|
|
||||||
state.buffers[bufnr] = nil
|
|
||||||
if buf_state and buf_state.timer then
|
|
||||||
buf_state.timer:stop()
|
|
||||||
buf_state.timer:close()
|
|
||||||
buf_state.timer = nil
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
---@private
|
|
||||||
function changetracking.reset(client_id)
|
|
||||||
local state = state_by_client[client_id]
|
|
||||||
if not state then
|
|
||||||
return
|
|
||||||
end
|
|
||||||
for _, buf_state in pairs(state.buffers) do
|
|
||||||
if buf_state.timer then
|
|
||||||
buf_state.timer:stop()
|
|
||||||
buf_state.timer:close()
|
|
||||||
buf_state.timer = nil
|
|
||||||
end
|
|
||||||
end
|
|
||||||
state.buffers = {}
|
|
||||||
end
|
|
||||||
|
|
||||||
---@private
|
|
||||||
--
|
|
||||||
-- Adjust debounce time by taking time of last didChange notification into
|
|
||||||
-- consideration. If the last didChange happened more than `debounce` time ago,
|
|
||||||
-- debounce can be skipped and otherwise maybe reduced.
|
|
||||||
--
|
|
||||||
-- This turns the debounce into a kind of client rate limiting
|
|
||||||
local function next_debounce(debounce, buf_state)
|
|
||||||
if debounce == 0 then
|
|
||||||
return 0
|
|
||||||
end
|
|
||||||
local ns_to_ms = 0.000001
|
|
||||||
if not buf_state.last_flush then
|
|
||||||
return debounce
|
|
||||||
end
|
|
||||||
local now = uv.hrtime()
|
|
||||||
local ms_since_last_flush = (now - buf_state.last_flush) * ns_to_ms
|
|
||||||
return math.max(debounce - ms_since_last_flush, 0)
|
|
||||||
end
|
|
||||||
|
|
||||||
---@private
|
|
||||||
function changetracking.prepare(bufnr, firstline, lastline, new_lastline)
|
|
||||||
local incremental_changes = function(client, buf_state)
|
|
||||||
local prev_lines = buf_state.lines
|
|
||||||
local curr_lines = buf_state.lines_tmp
|
|
||||||
|
|
||||||
local changed_lines = nvim_buf_get_lines(bufnr, firstline, new_lastline, true)
|
local changed_lines = nvim_buf_get_lines(bufnr, firstline, new_lastline, true)
|
||||||
for i = 1, firstline do
|
for i = 1, firstline do
|
||||||
@@ -466,12 +434,12 @@ do
|
|||||||
|
|
||||||
local line_ending = buf_get_line_ending(bufnr)
|
local line_ending = buf_get_line_ending(bufnr)
|
||||||
local incremental_change = sync.compute_diff(
|
local incremental_change = sync.compute_diff(
|
||||||
buf_state.lines,
|
state.lines,
|
||||||
curr_lines,
|
curr_lines,
|
||||||
firstline,
|
firstline,
|
||||||
lastline,
|
lastline,
|
||||||
new_lastline,
|
new_lastline,
|
||||||
client.offset_encoding or 'utf-16',
|
encoding,
|
||||||
line_ending
|
line_ending
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -483,44 +451,145 @@ do
|
|||||||
for i in ipairs(prev_lines) do
|
for i in ipairs(prev_lines) do
|
||||||
prev_lines[i] = nil
|
prev_lines[i] = nil
|
||||||
end
|
end
|
||||||
buf_state.lines = curr_lines
|
state.lines = curr_lines
|
||||||
buf_state.lines_tmp = prev_lines
|
state.lines_tmp = prev_lines
|
||||||
|
|
||||||
return incremental_change
|
return incremental_change
|
||||||
end
|
end
|
||||||
local full_changes = once(function()
|
|
||||||
return {
|
---@private
|
||||||
text = buf_get_full_text(bufnr),
|
function changetracking.init(client, bufnr)
|
||||||
|
assert(client.offset_encoding, 'lsp client must have an offset_encoding')
|
||||||
|
local group = get_group(client)
|
||||||
|
local state = state_by_group[group]
|
||||||
|
if state then
|
||||||
|
state.debounce = math.min(state.debounce, client.config.flags.debounce_text_changes or 150)
|
||||||
|
state.clients[client.id] = client
|
||||||
|
else
|
||||||
|
state = {
|
||||||
|
buffers = {},
|
||||||
|
debounce = client.config.flags.debounce_text_changes or 150,
|
||||||
|
clients = {
|
||||||
|
[client.id] = client,
|
||||||
|
},
|
||||||
}
|
}
|
||||||
end)
|
state_by_group[group] = state
|
||||||
local uri = vim.uri_from_bufnr(bufnr)
|
|
||||||
return function(client)
|
|
||||||
if
|
|
||||||
vim.tbl_get(client.server_capabilities, 'textDocumentSync', 'change')
|
|
||||||
== protocol.TextDocumentSyncKind.None
|
|
||||||
then
|
|
||||||
return
|
|
||||||
end
|
end
|
||||||
local state = state_by_client[client.id]
|
|
||||||
local buf_state = state.buffers[bufnr]
|
local buf_state = state.buffers[bufnr]
|
||||||
|
if buf_state then
|
||||||
|
buf_state.refs = buf_state.refs + 1
|
||||||
|
else
|
||||||
|
buf_state = {
|
||||||
|
name = api.nvim_buf_get_name(bufnr),
|
||||||
|
lines = {},
|
||||||
|
lines_tmp = {},
|
||||||
|
pending_changes = {},
|
||||||
|
needs_flush = false,
|
||||||
|
refs = 1,
|
||||||
|
}
|
||||||
|
state.buffers[bufnr] = buf_state
|
||||||
|
if group.sync_kind == protocol.TextDocumentSyncKind.Incremental then
|
||||||
|
buf_state.lines = nvim_buf_get_lines(bufnr, 0, -1, true)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
---@private
|
||||||
|
function changetracking._get_and_set_name(client, bufnr, name)
|
||||||
|
local state = state_by_group[get_group(client)] or {}
|
||||||
|
local buf_state = (state.buffers or {})[bufnr]
|
||||||
|
local old_name = buf_state.name
|
||||||
|
buf_state.name = name
|
||||||
|
return old_name
|
||||||
|
end
|
||||||
|
|
||||||
|
---@private
|
||||||
|
function changetracking.reset_buf(client, bufnr)
|
||||||
|
changetracking.flush(client, bufnr)
|
||||||
|
local state = state_by_group[get_group(client)]
|
||||||
|
if not state then
|
||||||
|
return
|
||||||
|
end
|
||||||
|
assert(state.buffers, 'CTGroupState must have buffers')
|
||||||
|
local buf_state = state.buffers[bufnr]
|
||||||
|
buf_state.refs = buf_state.refs - 1
|
||||||
|
assert(buf_state.refs >= 0, 'refcount on buffer state must not get negative')
|
||||||
|
if buf_state.refs == 0 then
|
||||||
|
state.buffers[bufnr] = nil
|
||||||
changetracking._reset_timer(buf_state)
|
changetracking._reset_timer(buf_state)
|
||||||
local debounce = next_debounce(state.debounce, buf_state)
|
|
||||||
if state.use_incremental_sync then
|
|
||||||
-- This must be done immediately and cannot be delayed
|
|
||||||
-- The contents would further change and startline/endline may no longer fit
|
|
||||||
table.insert(buf_state.pending_changes, incremental_changes(client, buf_state))
|
|
||||||
end
|
end
|
||||||
buf_state.pending_change = function()
|
end
|
||||||
if buf_state.pending_change == nil then
|
|
||||||
|
---@private
|
||||||
|
function changetracking.reset(client)
|
||||||
|
local state = state_by_group[get_group(client)]
|
||||||
|
if not state then
|
||||||
|
return
|
||||||
|
end
|
||||||
|
state.clients[client.id] = nil
|
||||||
|
if vim.tbl_count(state.clients) == 0 then
|
||||||
|
for _, buf_state in pairs(state.buffers) do
|
||||||
|
changetracking._reset_timer(buf_state)
|
||||||
|
end
|
||||||
|
state.buffers = {}
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
---@private
|
||||||
|
--
|
||||||
|
-- Adjust debounce time by taking time of last didChange notification into
|
||||||
|
-- consideration. If the last didChange happened more than `debounce` time ago,
|
||||||
|
-- debounce can be skipped and otherwise maybe reduced.
|
||||||
|
--
|
||||||
|
-- This turns the debounce into a kind of client rate limiting
|
||||||
|
--
|
||||||
|
---@param debounce number
|
||||||
|
---@param buf_state CTBufferState
|
||||||
|
---@return number
|
||||||
|
local function next_debounce(debounce, buf_state)
|
||||||
|
if debounce == 0 then
|
||||||
|
return 0
|
||||||
|
end
|
||||||
|
local ns_to_ms = 0.000001
|
||||||
|
if not buf_state.last_flush then
|
||||||
|
return debounce
|
||||||
|
end
|
||||||
|
local now = uv.hrtime()
|
||||||
|
local ms_since_last_flush = (now - buf_state.last_flush) * ns_to_ms
|
||||||
|
return math.max(debounce - ms_since_last_flush, 0)
|
||||||
|
end
|
||||||
|
|
||||||
|
---@private
|
||||||
|
---@param bufnr number
|
||||||
|
---@param sync_kind number protocol.TextDocumentSyncKind
|
||||||
|
---@param state CTGroupState
|
||||||
|
---@param buf_state CTBufferState
|
||||||
|
local function send_changes(bufnr, sync_kind, state, buf_state)
|
||||||
|
if not buf_state.needs_flush then
|
||||||
return
|
return
|
||||||
end
|
end
|
||||||
buf_state.pending_change = nil
|
|
||||||
buf_state.last_flush = uv.hrtime()
|
buf_state.last_flush = uv.hrtime()
|
||||||
if client.is_stopped() or not api.nvim_buf_is_valid(bufnr) then
|
buf_state.needs_flush = false
|
||||||
|
|
||||||
|
if not api.nvim_buf_is_valid(bufnr) then
|
||||||
|
buf_state.pending_changes = {}
|
||||||
return
|
return
|
||||||
end
|
end
|
||||||
local changes = state.use_incremental_sync and buf_state.pending_changes
|
|
||||||
or { full_changes() }
|
local changes
|
||||||
|
if sync_kind == protocol.TextDocumentSyncKind.None then
|
||||||
|
return
|
||||||
|
elseif sync_kind == protocol.TextDocumentSyncKind.Incremental then
|
||||||
|
changes = buf_state.pending_changes
|
||||||
|
buf_state.pending_changes = {}
|
||||||
|
else
|
||||||
|
changes = {
|
||||||
|
{ text = buf_get_full_text(bufnr) },
|
||||||
|
}
|
||||||
|
end
|
||||||
|
local uri = vim.uri_from_bufnr(bufnr)
|
||||||
|
for _, client in pairs(state.clients) do
|
||||||
|
if not client.is_stopped() and lsp.buf_is_attached(bufnr, client.id) then
|
||||||
client.notify('textDocument/didChange', {
|
client.notify('textDocument/didChange', {
|
||||||
textDocument = {
|
textDocument = {
|
||||||
uri = uri,
|
uri = uri,
|
||||||
@@ -528,46 +597,90 @@ do
|
|||||||
},
|
},
|
||||||
contentChanges = changes,
|
contentChanges = changes,
|
||||||
})
|
})
|
||||||
buf_state.pending_changes = {}
|
|
||||||
end
|
|
||||||
if debounce == 0 then
|
|
||||||
buf_state.pending_change()
|
|
||||||
else
|
|
||||||
local timer = uv.new_timer()
|
|
||||||
buf_state.timer = timer
|
|
||||||
-- Must use schedule_wrap because `full_changes()` calls nvim_buf_get_lines
|
|
||||||
timer:start(debounce, 0, vim.schedule_wrap(buf_state.pending_change))
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
---@private
|
||||||
|
function changetracking.send_changes(bufnr, firstline, lastline, new_lastline)
|
||||||
|
local groups = {}
|
||||||
|
for _, client in pairs(lsp.get_active_clients({ bufnr = bufnr })) do
|
||||||
|
local group = get_group(client)
|
||||||
|
groups[group_key(group)] = group
|
||||||
|
end
|
||||||
|
for _, group in pairs(groups) do
|
||||||
|
local state = state_by_group[group]
|
||||||
|
if not state then
|
||||||
|
error(
|
||||||
|
string.format(
|
||||||
|
'changetracking.init must have been called for all LSP clients. group=%s states=%s',
|
||||||
|
vim.inspect(group),
|
||||||
|
vim.inspect(vim.tbl_keys(state_by_group))
|
||||||
|
)
|
||||||
|
)
|
||||||
|
end
|
||||||
|
local buf_state = state.buffers[bufnr]
|
||||||
|
buf_state.needs_flush = true
|
||||||
|
changetracking._reset_timer(buf_state)
|
||||||
|
local debounce = next_debounce(state.debounce, buf_state)
|
||||||
|
if group.sync_kind == protocol.TextDocumentSyncKind.Incremental then
|
||||||
|
-- This must be done immediately and cannot be delayed
|
||||||
|
-- The contents would further change and startline/endline may no longer fit
|
||||||
|
local changes = incremental_changes(
|
||||||
|
buf_state,
|
||||||
|
group.offset_encoding,
|
||||||
|
bufnr,
|
||||||
|
firstline,
|
||||||
|
lastline,
|
||||||
|
new_lastline
|
||||||
|
)
|
||||||
|
table.insert(buf_state.pending_changes, changes)
|
||||||
|
end
|
||||||
|
if debounce == 0 then
|
||||||
|
send_changes(bufnr, group.sync_kind, state, buf_state)
|
||||||
|
else
|
||||||
|
local timer = uv.new_timer()
|
||||||
|
buf_state.timer = timer
|
||||||
|
timer:start(
|
||||||
|
debounce,
|
||||||
|
0,
|
||||||
|
vim.schedule_wrap(function()
|
||||||
|
changetracking._reset_timer(buf_state)
|
||||||
|
send_changes(bufnr, group.sync_kind, state, buf_state)
|
||||||
|
end)
|
||||||
|
)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
---@private
|
||||||
function changetracking._reset_timer(buf_state)
|
function changetracking._reset_timer(buf_state)
|
||||||
if buf_state.timer then
|
local timer = buf_state.timer
|
||||||
buf_state.timer:stop()
|
if timer then
|
||||||
buf_state.timer:close()
|
|
||||||
buf_state.timer = nil
|
buf_state.timer = nil
|
||||||
|
if not timer:is_closing() then
|
||||||
|
timer:stop()
|
||||||
|
timer:close()
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
--- Flushes any outstanding change notification.
|
--- Flushes any outstanding change notification.
|
||||||
---@private
|
---@private
|
||||||
function changetracking.flush(client, bufnr)
|
function changetracking.flush(client, bufnr)
|
||||||
local state = state_by_client[client.id]
|
local group = get_group(client)
|
||||||
|
local state = state_by_group[group]
|
||||||
if not state then
|
if not state then
|
||||||
return
|
return
|
||||||
end
|
end
|
||||||
if bufnr then
|
if bufnr then
|
||||||
local buf_state = state.buffers[bufnr] or {}
|
local buf_state = state.buffers[bufnr] or {}
|
||||||
changetracking._reset_timer(buf_state)
|
changetracking._reset_timer(buf_state)
|
||||||
if buf_state.pending_change then
|
send_changes(bufnr, group.sync_kind, state, buf_state)
|
||||||
buf_state.pending_change()
|
|
||||||
end
|
|
||||||
else
|
else
|
||||||
for _, buf_state in pairs(state.buffers) do
|
for buf, buf_state in pairs(state.buffers) do
|
||||||
changetracking._reset_timer(buf_state)
|
changetracking._reset_timer(buf_state)
|
||||||
if buf_state.pending_change then
|
send_changes(buf, group.sync_kind, state, buf_state)
|
||||||
buf_state.pending_change()
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -1030,11 +1143,12 @@ function lsp.start_client(config)
|
|||||||
end)
|
end)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
local client = active_clients[client_id] and active_clients[client_id]
|
||||||
|
or uninitialized_clients[client_id]
|
||||||
active_clients[client_id] = nil
|
active_clients[client_id] = nil
|
||||||
uninitialized_clients[client_id] = nil
|
uninitialized_clients[client_id] = nil
|
||||||
|
|
||||||
changetracking.reset(client_id)
|
changetracking.reset(client)
|
||||||
if code ~= 0 or (signal ~= 0 and signal ~= 15) then
|
if code ~= 0 or (signal ~= 0 and signal ~= 15) then
|
||||||
local msg =
|
local msg =
|
||||||
string.format('Client %s quit with exit code %s and signal %s', client_id, code, signal)
|
string.format('Client %s quit with exit code %s and signal %s', client_id, code, signal)
|
||||||
@@ -1414,9 +1528,7 @@ do
|
|||||||
return true
|
return true
|
||||||
end
|
end
|
||||||
util.buf_versions[bufnr] = changedtick
|
util.buf_versions[bufnr] = changedtick
|
||||||
local compute_change_and_notify =
|
changetracking.send_changes(bufnr, firstline, lastline, new_lastline)
|
||||||
changetracking.prepare(bufnr, firstline, lastline, new_lastline)
|
|
||||||
for_each_buffer_client(bufnr, compute_change_and_notify)
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user