mirror of
https://github.com/neovim/neovim.git
synced 2025-01-01 17:23:36 -07:00
123 lines
3.7 KiB
Lua
123 lines
3.7 KiB
Lua
local a = vim.api
|
|
|
|
-- support reload for quick experimentation
|
|
local TSHighlighter = rawget(vim.treesitter, 'TSHighlighter') or {}
|
|
TSHighlighter.__index = TSHighlighter
|
|
|
|
-- These are conventions defined by tree-sitter, though it
|
|
-- needs to be user extensible also.
|
|
-- TODO(bfredl): this is very much incomplete, we will need to
|
|
-- go through a few tree-sitter provided queries and decide
|
|
-- on translations that makes the most sense.
|
|
TSHighlighter.hl_map = {
|
|
keyword="Keyword",
|
|
string="String",
|
|
type="Type",
|
|
comment="Comment",
|
|
constant="Constant",
|
|
operator="Operator",
|
|
number="Number",
|
|
label="Label",
|
|
["function"]="Function",
|
|
["function.special"]="Function",
|
|
}
|
|
|
|
function TSHighlighter.new(query, bufnr, ft)
|
|
local self = setmetatable({}, TSHighlighter)
|
|
self.parser = vim.treesitter.get_parser(bufnr, ft, function(...) self:on_change(...) end)
|
|
self.buf = self.parser.bufnr
|
|
-- TODO(bfredl): perhaps on_start should be called uncondionally, instead for only on mod?
|
|
local tree = self.parser:parse()
|
|
self.root = tree:root()
|
|
self:set_query(query)
|
|
self.edit_count = 0
|
|
self.redraw_count = 0
|
|
self.line_count = {}
|
|
a.nvim_buf_set_option(self.buf, "syntax", "")
|
|
a.nvim__buf_set_luahl(self.buf, {
|
|
on_start=function(...) return self:on_start(...) end,
|
|
on_window=function(...) return self:on_window(...) end,
|
|
on_line=function(...) return self:on_line(...) end,
|
|
})
|
|
|
|
-- Tricky: if syntax hasn't been enabled, we need to reload color scheme
|
|
-- but use synload.vim rather than syntax.vim to not enable
|
|
-- syntax FileType autocmds. Later on we should integrate with the
|
|
-- `:syntax` and `set syntax=...` machinery properly.
|
|
if vim.g.syntax_on ~= 1 then
|
|
vim.api.nvim_command("runtime! syntax/synload.vim")
|
|
end
|
|
return self
|
|
end
|
|
|
|
function TSHighlighter:set_query(query)
|
|
if type(query) == "string" then
|
|
query = vim.treesitter.parse_query(self.parser.lang, query)
|
|
end
|
|
self.query = query
|
|
|
|
self.id_map = {}
|
|
for i, capture in ipairs(self.query.captures) do
|
|
local hl = 0
|
|
local firstc = string.sub(capture, 1, 1)
|
|
local hl_group = self.hl_map[capture]
|
|
if firstc ~= string.lower(firstc) then
|
|
hl_group = vim.split(capture, '.', true)[1]
|
|
end
|
|
if hl_group then
|
|
hl = a.nvim_get_hl_id_by_name(hl_group)
|
|
end
|
|
self.id_map[i] = hl
|
|
end
|
|
|
|
a.nvim__buf_redraw_range(self.buf, 0, a.nvim_buf_line_count(self.buf))
|
|
end
|
|
|
|
function TSHighlighter:on_change(changes)
|
|
for _, ch in ipairs(changes or {}) do
|
|
a.nvim__buf_redraw_range(self.buf, ch[1], ch[3]+1)
|
|
end
|
|
self.edit_count = self.edit_count + 1
|
|
end
|
|
|
|
function TSHighlighter:on_start(_, _buf, _tick)
|
|
local tree = self.parser:parse()
|
|
self.root = tree:root()
|
|
end
|
|
|
|
function TSHighlighter:on_window(_, _win, _buf, _topline, botline)
|
|
self.iter = nil
|
|
self.nextrow = 0
|
|
self.botline = botline
|
|
self.redraw_count = self.redraw_count + 1
|
|
end
|
|
|
|
function TSHighlighter:on_line(_, _win, buf, line)
|
|
if self.iter == nil then
|
|
self.iter = self.query:iter_captures(self.root,buf,line,self.botline)
|
|
end
|
|
while line >= self.nextrow do
|
|
local capture, node, match = self.iter()
|
|
local active = true
|
|
if capture == nil then
|
|
break
|
|
end
|
|
if match ~= nil then
|
|
active = self:run_pred(match)
|
|
match.active = active
|
|
end
|
|
local start_row, start_col, end_row, end_col = node:range()
|
|
local hl = self.id_map[capture]
|
|
if hl > 0 and active and end_row >= line then
|
|
a.nvim__put_attr(hl, start_row, start_col, end_row, end_col)
|
|
end
|
|
if start_row > line then
|
|
self.nextrow = start_row
|
|
end
|
|
end
|
|
self.line_count[line] = (self.line_count[line] or 0) + 1
|
|
--return tostring(self.line_count[line])
|
|
end
|
|
|
|
return TSHighlighter
|