Correctly handle on_reload, on_detach, refactor indexing functions

This commit is contained in:
Dmytro Meleshko 2021-11-07 13:45:43 +02:00
parent 2d85e76c72
commit e2c3b009b6
2 changed files with 114 additions and 45 deletions

View File

@ -3,10 +3,13 @@
---@field public regex any
---@field public length number
---@field public pattern string
---@field public indexing_chunk_size number
---@field public indexing_interval number
---@field public timer any|nil
---@field public lines_count number
---@field public lines_words table<number, string[]>
---@field public processing boolean
---@field public closed boolean
---@field public on_close_cb fun()|nil
local buffer = {}
---Create new buffer object
@ -20,50 +23,78 @@ function buffer.new(bufnr, length, pattern)
self.regex = vim.regex(pattern)
self.length = length
self.pattern = pattern
self.indexing_chunk_size = 1000
self.indexing_interval = 200
self.timer = nil
self.lines_count = 0
self.lines_words = {}
self.processing = false
self.closed = false
self.on_close_cb = nil
return self
end
---Close buffer
function buffer.close(self)
if self.timer then
self.closed = true
self:stop_indexing_timer()
self.lines_count = 0
self.lines_words = {}
if self.on_close_cb then
self.on_close_cb()
end
end
function buffer.stop_indexing_timer(self)
if self.timer and not self.timer:is_closing() then
self.timer:stop()
self.timer:close()
self.timer = nil
end
self.lines_words = {}
self.lines_count = 0
self.timer = nil
end
---Indexing buffer
function buffer.index(self)
self.processing = true
self.lines_count = vim.api.nvim_buf_line_count(self.bufnr)
local index = 1
local lines = vim.api.nvim_buf_get_lines(self.bufnr, 0, -1, false)
for i = 1, self.lines_count do
self.lines_words[i] = {}
end
self:index_range_async(0, self.lines_count)
end
function buffer.index_range(self, range_start, range_end)
vim.api.nvim_buf_call(self.bufnr, function()
local lines = vim.api.nvim_buf_get_lines(self.bufnr, range_start, range_end, true)
for i, line in ipairs(lines) do
self:index_line(range_start + i, line)
end
end)
end
function buffer.index_range_async(self, range_start, range_end)
local chunk_start = range_start
local lines = vim.api.nvim_buf_get_lines(self.bufnr, range_start, range_end, true)
self.timer = vim.loop.new_timer()
self.timer:start(
0,
200,
self.indexing_interval,
vim.schedule_wrap(function()
local chunk = math.min(index + 1000, #lines)
if self.closed then
return
end
local chunk_end = math.min(chunk_start + self.indexing_chunk_size, range_end)
vim.api.nvim_buf_call(self.bufnr, function()
for i = index, chunk do
self:index_line(i, lines[i] or '')
for linenr = chunk_start + 1, chunk_end do
self:index_line(linenr, lines[linenr])
end
end)
index = chunk + 1
chunk_start = chunk_end
if chunk >= #lines then
if self.timer then
self.timer:stop()
self.timer:close()
self.timer = nil
end
self.processing = false
if chunk_end >= range_end then
self:stop_indexing_timer()
end
end)
)
@ -71,23 +102,31 @@ end
--- watch
function buffer.watch(self)
-- NOTE: As far as I know, indexing in watching can't be done asynchronously
-- because even built-in commands generate multiple consequent `on_lines`
-- events, and I'm not even mentioning plugins here. To get accurate results
-- we would have to either re-index the entire file on throttled events (slow
-- and looses the benefit of on_lines watching), or put the events in a
-- queue, which would complicate the plugin a lot. Plus, most changes which
-- trigger this event will be from regular editing, and so 99% of the time
-- they will affect only 1-2 lines.
vim.api.nvim_buf_attach(self.bufnr, false, {
-- NOTE: line indexes are 0-based and the last line is not inclusive.
on_lines = function(_, _, _, first_line, old_last_line, new_last_line, _, _, _)
if not vim.api.nvim_buf_is_valid(self.bufnr) then
self:close()
if self.closed then
return true
end
local delta = new_last_line - old_last_line
local new_lines_count = self.lines_count + delta
if new_lines_count == 0 then -- clear
local old_lines_count = self.lines_count
local new_lines_count = old_lines_count + delta
if new_lines_count == 0 then -- clear
-- This branch protects against bugs after full-file deletion. If you
-- do, for example, gdGG, the new_last_line of the event will be zero.
-- Which is not true, a buffer always contains at least one empty line,
-- only unloaded buffers contain zero lines.
new_lines_count = 1
for i = self.lines_count, 2, -1 do
for i = old_lines_count, 2, -1 do
self.lines_words[i] = nil
end
self.lines_words[1] = {}
@ -96,11 +135,11 @@ function buffer.watch(self)
-- all of them will be filled in the next loop, but in reverse order
-- (which is why I am concerned about preallocation). Why is there no
-- built-in function to do this in Lua???
for i = self.lines_count + 1, new_lines_count do
for i = old_lines_count + 1, new_lines_count do
self.lines_words[i] = vim.NIL
end
-- Move forwards the unchanged elements in the tail part.
for i = self.lines_count, old_last_line + 1, -1 do
for i = old_lines_count, old_last_line + 1, -1 do
self.lines_words[i + delta] = self.lines_words[i]
end
-- Fill in new tables for the added lines.
@ -109,26 +148,48 @@ function buffer.watch(self)
end
elseif delta < 0 then -- remove
-- Move backwards the unchanged elements in the tail part.
for i = old_last_line + 1, self.lines_count do
for i = old_last_line + 1, old_lines_count do
self.lines_words[i + delta] = self.lines_words[i]
end
-- Remove (already copied) tables from the end, in reverse order, so
-- that we don't make holes in the lines table.
for i = self.lines_count, new_lines_count + 1, -1 do
for i = old_lines_count, new_lines_count + 1, -1 do
self.lines_words[i] = nil
end
end
self.lines_count = new_lines_count
-- replace lines
local lines = vim.api.nvim_buf_get_lines(self.bufnr, first_line, new_last_line, true)
vim.api.nvim_buf_call(self.bufnr, function()
for i, line in ipairs(lines) do
if line then
self:index_line(first_line + i, line)
end
self:index_range(first_line, new_last_line)
end,
on_reload = function(_, _)
if self.closed then
return true
end
-- The logic for adjusting lines list on buffer reloads is much simpler
-- because tables of all lines can be assumed to be fresh.
local new_lines_count = vim.api.nvim_buf_line_count(self.bufnr)
if new_lines_count > self.lines_count then -- append
for i = self.lines_count + 1, new_lines_count do
self.lines_words[i] = {}
end
end)
elseif new_lines_count < self.lines_count then -- remove
for i = self.lines_count, new_lines_count + 1, -1 do
self.lines_words[i] = nil
end
end
self.lines_count = new_lines_count
self:index_range(0, self.lines_count)
end,
on_detach = function(_, _)
if self.closed then
return true
end
self:close()
end,
})
end
@ -136,7 +197,10 @@ end
---@param linenr number
---@param line string
function buffer.index_line(self, linenr, line)
local words = {}
local words = self.lines_words[linenr]
for k, _ in ipairs(words) do
words[k] = nil
end
local word_i = 1
local remaining = line
@ -155,8 +219,6 @@ function buffer.index_line(self, linenr, line)
break
end
end
self.lines_words[linenr] = words
end
--- get_words

View File

@ -34,15 +34,19 @@ source.complete = function(self, params, callback)
})
local processing = false
for _, buf in ipairs(self:_get_buffers(params)) do
processing = processing or buf.processing
local bufs = self:_get_buffers(params)
for _, buf in ipairs(bufs) do
if buf.timer then
processing = true
break
end
end
vim.defer_fn(vim.schedule_wrap(function()
vim.defer_fn(function()
local input = string.sub(params.context.cursor_before_line, params.offset)
local items = {}
local words = {}
for _, buf in ipairs(self:_get_buffers(params)) do
for _, buf in ipairs(bufs) do
for _, word in ipairs(buf:get_words()) do
if not words[word] and input ~= word then
words[word] = true
@ -58,7 +62,7 @@ source.complete = function(self, params, callback)
items = items,
isIncomplete = processing,
})
end), processing and 100 or 0)
end, processing and 100 or 0)
end
--- _get_bufs
@ -71,6 +75,9 @@ source._get_buffers = function(self, params)
params.option.keyword_length,
params.option.keyword_pattern
)
new_buf.on_close_cb = function()
self.buffers[bufnr] = nil
end
new_buf:index()
new_buf:watch()
self.buffers[bufnr] = new_buf