2019-06-07 05:21:00 -07:00
|
|
|
local a = vim.api
|
|
|
|
|
2019-06-07 09:19:59 -07:00
|
|
|
local Parser = {}
|
|
|
|
Parser.__index = Parser
|
|
|
|
|
2019-06-17 12:46:31 -07:00
|
|
|
-- TODO(bfredl): currently we retain parsers for the lifetime of the buffer.
|
|
|
|
-- Consider use weak references to release parser if all plugins are done with
|
|
|
|
-- it.
|
|
|
|
local parsers = {}
|
|
|
|
|
|
|
|
function Parser:parse()
|
|
|
|
if self.valid then
|
2019-06-07 09:19:59 -07:00
|
|
|
return self.tree
|
2019-06-07 05:21:00 -07:00
|
|
|
end
|
2019-06-07 09:19:59 -07:00
|
|
|
self.tree = self._parser:parse_buf(self.bufnr)
|
|
|
|
self.valid = true
|
|
|
|
return self.tree
|
2019-06-07 05:21:00 -07:00
|
|
|
end
|
|
|
|
|
2019-06-17 12:46:31 -07:00
|
|
|
local function on_lines(self, bufnr, _, start_row, oldstopline, stop_row)
|
2019-06-07 05:21:00 -07:00
|
|
|
local start_byte = a.nvim_buf_get_offset(bufnr,start_row)
|
|
|
|
-- a bit messy, should we expose edited but not reparsed tree?
|
|
|
|
-- are multiple edits safe in general?
|
2019-06-07 09:19:59 -07:00
|
|
|
local root = self._parser:tree():root()
|
2019-06-07 05:21:00 -07:00
|
|
|
-- TODO: add proper lookup function!
|
|
|
|
local inode = root:descendant_for_point_range(oldstopline+9000,0, oldstopline,0)
|
|
|
|
if inode == nil then
|
|
|
|
local stop_byte = a.nvim_buf_get_offset(bufnr,stop_row)
|
2019-06-17 12:46:31 -07:00
|
|
|
self._parser:edit(start_byte,stop_byte,stop_byte,
|
|
|
|
start_row,0,stop_row,0,stop_row,0)
|
2019-06-07 05:21:00 -07:00
|
|
|
else
|
|
|
|
local fakeoldstoprow, fakeoldstopcol, fakebyteoldstop = inode:start()
|
|
|
|
local fake_rows = fakeoldstoprow-oldstopline
|
|
|
|
local fakestop = stop_row+fake_rows
|
|
|
|
local fakebytestop = a.nvim_buf_get_offset(bufnr,fakestop)+fakeoldstopcol
|
2019-06-17 12:46:31 -07:00
|
|
|
self._parser:edit(start_byte, fakebyteoldstop, fakebytestop,
|
|
|
|
start_row, 0,
|
|
|
|
fakeoldstoprow, fakeoldstopcol,
|
|
|
|
fakestop, fakeoldstopcol)
|
2019-06-07 05:21:00 -07:00
|
|
|
end
|
2019-06-07 09:19:59 -07:00
|
|
|
self.valid = false
|
2019-06-07 05:21:00 -07:00
|
|
|
end
|
|
|
|
|
2019-06-17 12:46:31 -07:00
|
|
|
local function create_parser(bufnr, ft, id)
|
2019-06-07 05:21:00 -07:00
|
|
|
if bufnr == 0 then
|
|
|
|
bufnr = a.nvim_get_current_buf()
|
|
|
|
end
|
2019-06-07 09:19:59 -07:00
|
|
|
local self = setmetatable({bufnr=bufnr, valid=false}, Parser)
|
2019-06-08 06:51:38 -07:00
|
|
|
self._parser = vim._create_ts_parser(ft)
|
2019-06-17 12:46:31 -07:00
|
|
|
self:parse()
|
2019-06-07 09:19:59 -07:00
|
|
|
-- TODO: use weakref to self, so that the parser is free'd is no plugin is
|
|
|
|
-- using it.
|
2019-06-17 12:46:31 -07:00
|
|
|
local function lines_cb(ev, ...)
|
|
|
|
return on_lines(self, ...)
|
2019-06-07 05:21:00 -07:00
|
|
|
end
|
2019-06-17 12:46:31 -07:00
|
|
|
local detach_cb = nil
|
|
|
|
if id ~= nil then
|
|
|
|
detach_cb = function()
|
|
|
|
if parsers[id] == self then
|
|
|
|
parsers[id] = nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
a.nvim_buf_attach(self.bufnr, false, {on_lines=lines_cb, on_detach=detach_cb})
|
2019-06-07 09:19:59 -07:00
|
|
|
return self
|
2019-06-07 05:21:00 -07:00
|
|
|
end
|
|
|
|
|
2019-06-17 12:46:31 -07:00
|
|
|
local function get_parser(bufnr, ft)
|
|
|
|
if bufnr == nil or bufnr == 0 then
|
|
|
|
bufnr = a.nvim_get_current_buf()
|
|
|
|
end
|
|
|
|
if ft == nil then
|
|
|
|
ft = a.nvim_buf_get_option(bufnr, "filetype")
|
|
|
|
end
|
|
|
|
local id = tostring(bufnr)..'_'..ft
|
|
|
|
|
|
|
|
if parsers[id] == nil then
|
|
|
|
parsers[id] = create_parser(bufnr, ft, id)
|
|
|
|
end
|
|
|
|
return parsers[id]
|
|
|
|
end
|
2019-06-07 09:19:59 -07:00
|
|
|
|
2019-06-15 05:05:35 -07:00
|
|
|
return {
|
2019-06-17 12:46:31 -07:00
|
|
|
get_parser=get_parser,
|
2019-06-15 05:05:35 -07:00
|
|
|
create_parser=create_parser,
|
|
|
|
add_language=vim._ts_add_language,
|
|
|
|
inspect_language=vim._ts_inspect_language,
|
|
|
|
}
|