mirror of
https://github.com/neovim/neovim.git
synced 2025-01-01 17:23:36 -07:00
1fb44ba835
Treesitter changed their decoders and apparently thus causing this change. This decoder change happened on ee9a3c0ebb218990cf391ed987be7f2448c54a73.
233 lines
5.9 KiB
Lua
233 lines
5.9 KiB
Lua
local a = vim.api
|
|
|
|
-- 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
|
|
|
|
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
|
|
for _, cb in ipairs(self.change_cbs) do
|
|
cb(changes)
|
|
end
|
|
return self.tree, changes
|
|
end
|
|
|
|
function Parser:_on_lines(bufnr, _, 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
|
|
end
|
|
|
|
local M = {
|
|
parse_query = vim._ts_parse_query,
|
|
}
|
|
|
|
setmetatable(M, {
|
|
__index = function (t, k)
|
|
if k == "TSHighlighter" then
|
|
t[k] = require'vim.tshighlighter'
|
|
return t[k]
|
|
end
|
|
end
|
|
})
|
|
|
|
function M.require_language(lang, path)
|
|
if vim._ts_has_language(lang) then
|
|
return true
|
|
end
|
|
if path == nil then
|
|
local fname = 'parser/' .. lang .. '.*'
|
|
local paths = a.nvim_get_runtime_file(fname, false)
|
|
if #paths == 0 then
|
|
-- TODO(bfredl): help tag?
|
|
error("no parser for '"..lang.."' language")
|
|
end
|
|
path = paths[1]
|
|
end
|
|
vim._ts_add_language(path, lang)
|
|
end
|
|
|
|
function M.inspect_language(lang)
|
|
M.require_language(lang)
|
|
return vim._ts_inspect_language(lang)
|
|
end
|
|
|
|
function M.create_parser(bufnr, lang, id)
|
|
M.require_language(lang)
|
|
if bufnr == 0 then
|
|
bufnr = a.nvim_get_current_buf()
|
|
end
|
|
local self = setmetatable({bufnr=bufnr, lang=lang, valid=false}, Parser)
|
|
self._parser = vim._create_ts_parser(lang)
|
|
self.change_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
|
|
|
|
function M.get_parser(bufnr, ft, cb)
|
|
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] = M.create_parser(bufnr, ft, id)
|
|
end
|
|
if cb ~= nil then
|
|
table.insert(parsers[id].change_cbs, cb)
|
|
end
|
|
return parsers[id]
|
|
end
|
|
|
|
-- query: pattern matching on trees
|
|
-- predicate matching is implemented in lua
|
|
local Query = {}
|
|
Query.__index = Query
|
|
|
|
local magic_prefixes = {['\\v']=true, ['\\m']=true, ['\\M']=true, ['\\V']=true}
|
|
local function check_magic(str)
|
|
if string.len(str) < 2 or magic_prefixes[string.sub(str,1,2)] then
|
|
return str
|
|
end
|
|
return '\\v'..str
|
|
end
|
|
|
|
function M.parse_query(lang, query)
|
|
M.require_language(lang)
|
|
local self = setmetatable({}, Query)
|
|
self.query = vim._ts_parse_query(lang, vim.fn.escape(query,'\\'))
|
|
self.info = self.query:inspect()
|
|
self.captures = self.info.captures
|
|
self.regexes = {}
|
|
for id,preds in pairs(self.info.patterns) do
|
|
local regexes = {}
|
|
for i, pred in ipairs(preds) do
|
|
if (pred[1] == "match?" and type(pred[2]) == "number"
|
|
and type(pred[3]) == "string") then
|
|
regexes[i] = vim.regex(check_magic(pred[3]))
|
|
end
|
|
end
|
|
if next(regexes) then
|
|
self.regexes[id] = regexes
|
|
end
|
|
end
|
|
return self
|
|
end
|
|
|
|
local function get_node_text(node, bufnr)
|
|
local start_row, start_col, end_row, end_col = node:range()
|
|
if start_row ~= end_row then
|
|
return nil
|
|
end
|
|
local line = a.nvim_buf_get_lines(bufnr, start_row, start_row+1, true)[1]
|
|
return string.sub(line, start_col+1, end_col)
|
|
end
|
|
|
|
function Query:match_preds(match, pattern, bufnr)
|
|
local preds = self.info.patterns[pattern]
|
|
if not preds then
|
|
return true
|
|
end
|
|
local regexes = self.regexes[pattern]
|
|
for i, pred in pairs(preds) do
|
|
if pred[1] == "eq?" then
|
|
local node = match[pred[2]]
|
|
local node_text = get_node_text(node, bufnr)
|
|
|
|
local str
|
|
if type(pred[3]) == "string" then
|
|
-- (eq? @aa "foo")
|
|
str = pred[3]
|
|
else
|
|
-- (eq? @aa @bb)
|
|
str = get_node_text(match[pred[3]], bufnr)
|
|
end
|
|
|
|
if node_text ~= str or str == nil then
|
|
return false
|
|
end
|
|
elseif pred[1] == "match?" then
|
|
if not regexes or not regexes[i] then
|
|
return false
|
|
end
|
|
local node = match[pred[2]]
|
|
local start_row, start_col, end_row, end_col = node:range()
|
|
if start_row ~= end_row then
|
|
return false
|
|
end
|
|
return regexes[i]:match_line(bufnr, start_row, start_col, end_col)
|
|
else
|
|
return false
|
|
end
|
|
end
|
|
return true
|
|
end
|
|
|
|
function Query:iter_captures(node, bufnr, start, stop)
|
|
if bufnr == 0 then
|
|
bufnr = vim.api.nvim_get_current_buf()
|
|
end
|
|
local raw_iter = node:_rawquery(self.query,true,start,stop)
|
|
local function iter()
|
|
local capture, captured_node, match = raw_iter()
|
|
if match ~= nil then
|
|
local active = self:match_preds(match, match.pattern, bufnr)
|
|
match.active = active
|
|
if not active then
|
|
return iter() -- tail call: try next match
|
|
end
|
|
end
|
|
return capture, captured_node
|
|
end
|
|
return iter
|
|
end
|
|
|
|
function Query:iter_matches(node, bufnr, start, stop)
|
|
if bufnr == 0 then
|
|
bufnr = vim.api.nvim_get_current_buf()
|
|
end
|
|
local raw_iter = node:_rawquery(self.query,false,start,stop)
|
|
local function iter()
|
|
local pattern, match = raw_iter()
|
|
if match ~= nil then
|
|
local active = self:match_preds(match, pattern, bufnr)
|
|
if not active then
|
|
return iter() -- tail call: try next match
|
|
end
|
|
end
|
|
return pattern, match
|
|
end
|
|
return iter
|
|
end
|
|
|
|
return M
|