mirror of
https://github.com/neovim/neovim.git
synced 2025-09-06 11:28:22 +00:00
149 lines
4.4 KiB
Lua
149 lines
4.4 KiB
Lua
local a = vim.api
|
|
local query = require'vim.treesitter.query'
|
|
local language = require'vim.treesitter.language'
|
|
|
|
-- 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 = {}
|
|
|
|
local Parser = {}
|
|
Parser.__index = Parser
|
|
|
|
--- Parses the buffer if needed and returns a tree.
|
|
--
|
|
-- Calling this will call the on_changedtree callbacks if the tree has changed.
|
|
--
|
|
-- @returns An up to date tree
|
|
-- @returns If the tree changed with this call, the changed ranges
|
|
function Parser:parse()
|
|
if self.valid then
|
|
return self.tree
|
|
end
|
|
local changes
|
|
self.tree, changes = self._parser:parse_buf(self.bufnr)
|
|
self.valid = true
|
|
|
|
if not vim.tbl_isempty(changes) then
|
|
for _, cb in ipairs(self.changedtree_cbs) do
|
|
cb(changes)
|
|
end
|
|
end
|
|
|
|
return self.tree, changes
|
|
end
|
|
|
|
function Parser:_on_lines(bufnr, changed_tick, start_row, old_stop_row, stop_row, old_byte_size)
|
|
local start_byte = a.nvim_buf_get_offset(bufnr,start_row)
|
|
local stop_byte = a.nvim_buf_get_offset(bufnr,stop_row)
|
|
local old_stop_byte = start_byte + old_byte_size
|
|
self._parser:edit(start_byte,old_stop_byte,stop_byte,
|
|
start_row,0,old_stop_row,0,stop_row,0)
|
|
self.valid = false
|
|
|
|
for _, cb in ipairs(self.lines_cbs) do
|
|
cb(bufnr, changed_tick, start_row, old_stop_row, stop_row, old_byte_size)
|
|
end
|
|
end
|
|
|
|
--- Sets the included ranges for the current parser
|
|
--
|
|
-- @param ranges A table of nodes that will be used as the ranges the parser should include.
|
|
function Parser:set_included_ranges(ranges)
|
|
self._parser:set_included_ranges(ranges)
|
|
-- The buffer will need to be parsed again later
|
|
self.valid = false
|
|
end
|
|
|
|
local M = vim.tbl_extend("error", query, language)
|
|
|
|
setmetatable(M, {
|
|
__index = function (t, k)
|
|
if k == "TSHighlighter" then
|
|
a.nvim_err_writeln("vim.TSHighlighter is deprecated, please use vim.treesitter.highlighter")
|
|
t[k] = require'vim.treesitter.highlighter'
|
|
return t[k]
|
|
elseif k == "highlighter" then
|
|
t[k] = require'vim.treesitter.highlighter'
|
|
return t[k]
|
|
end
|
|
end
|
|
})
|
|
|
|
--- Creates a new parser.
|
|
--
|
|
-- It is not recommended to use this, use vim.treesitter.get_parser() instead.
|
|
--
|
|
-- @param bufnr The buffer the parser will be tied to
|
|
-- @param lang The language of the parser.
|
|
-- @param id The id the parser will have
|
|
function M._create_parser(bufnr, lang, id)
|
|
language.require_language(lang)
|
|
if bufnr == 0 then
|
|
bufnr = a.nvim_get_current_buf()
|
|
end
|
|
|
|
vim.fn.bufload(bufnr)
|
|
|
|
local self = setmetatable({bufnr=bufnr, lang=lang, valid=false}, Parser)
|
|
self._parser = vim._create_ts_parser(lang)
|
|
self.changedtree_cbs = {}
|
|
self.lines_cbs = {}
|
|
self:parse()
|
|
-- TODO(bfredl): use weakref to self, so that the parser is free'd is no plugin is
|
|
-- using it.
|
|
local function lines_cb(_, ...)
|
|
return self:_on_lines(...)
|
|
end
|
|
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})
|
|
return self
|
|
end
|
|
|
|
--- Gets the parser for this bufnr / ft combination.
|
|
--
|
|
-- If needed this will create the parser.
|
|
-- Unconditionnally attach the provided callback
|
|
--
|
|
-- @param bufnr The buffer the parser should be tied to
|
|
-- @param ft The filetype of this parser
|
|
-- @param buf_attach_cbs An `nvim_buf_attach`-like table argument with the following keys :
|
|
-- `on_lines` : see `nvim_buf_attach`, but this will be called _after_ the parsers callback.
|
|
-- `on_changedtree` : a callback that will be called everytime the tree has syntactical changes.
|
|
-- it will only be passed one argument, that is a table of the ranges (as node ranges) that
|
|
-- changed.
|
|
--
|
|
-- @returns The parser
|
|
function M.get_parser(bufnr, lang, buf_attach_cbs)
|
|
if bufnr == nil or bufnr == 0 then
|
|
bufnr = a.nvim_get_current_buf()
|
|
end
|
|
if lang == nil then
|
|
lang = a.nvim_buf_get_option(bufnr, "filetype")
|
|
end
|
|
local id = tostring(bufnr)..'_'..lang
|
|
|
|
if parsers[id] == nil then
|
|
parsers[id] = M._create_parser(bufnr, lang, id)
|
|
end
|
|
|
|
if buf_attach_cbs and buf_attach_cbs.on_changedtree then
|
|
table.insert(parsers[id].changedtree_cbs, buf_attach_cbs.on_changedtree)
|
|
end
|
|
|
|
if buf_attach_cbs and buf_attach_cbs.on_lines then
|
|
table.insert(parsers[id].lines_cbs, buf_attach_cbs.on_lines)
|
|
end
|
|
|
|
return parsers[id]
|
|
end
|
|
|
|
return M
|