diff options
| author | Thomas Vigouroux <tomvig38@gmail.com> | 2020-04-19 09:45:54 +0200 |
|---|---|---|
| committer | Thomas Vigouroux <tomvig38@gmail.com> | 2020-04-19 09:45:54 +0200 |
| commit | 2526baf4cc5ea540188345c856adb162ce102674 (patch) | |
| tree | 6bdf2bb238354161c484e92e575a30a7d77b7d09 /lua/nvim-treesitter/query.lua | |
| parent | 5897d72b075991e4de5876c918be186d4551cf12 (diff) | |
feat: add parsers module and better match iter
The `parsers` module manages parsers for us, for now only in a really
basic way.
iter_prepared_mathes iters on an enhanced versions of the matches, where
captures are directly accessible via their names to allow things like :
((itentifier) @def.first (identifier) @def.last)
To be handled like this in lua:
match.def.first
match.def.last
Also adds a `set!` predicate to allow setting data within the prepared
match (see queries/lua/locals.scm) for examples.
Diffstat (limited to 'lua/nvim-treesitter/query.lua')
| -rw-r--r-- | lua/nvim-treesitter/query.lua | 76 |
1 files changed, 76 insertions, 0 deletions
diff --git a/lua/nvim-treesitter/query.lua b/lua/nvim-treesitter/query.lua new file mode 100644 index 00000000..be650c79 --- /dev/null +++ b/lua/nvim-treesitter/query.lua @@ -0,0 +1,76 @@ +-- Treesitter utils + +local api = vim.api +local ts = vim.treesitter + +local M = {} + +local function read_query_file(fname) + return table.concat(vim.fn.readfile(fname), '\n') +end + +function M.get_query(ft, query_name) + local query_files = api.nvim_get_runtime_file(string.format('queries/%s/%s.scm', ft, query_name), false) + if #query_files > 0 then + return ts.parse_query(ft, read_query_file(query_files[1])) + end +end + +function M.iter_prepared_matches(query, qnode, bufnr, start_row, end_row) + -- A function that splits a string on '.' + local function split(string) + local t = {} + for str in string.gmatch(string, "([^.]+)") do + table.insert(t, str) + end + + return t + end + + -- Given a path (i.e. a List(String)) this functions inserts value at path + local function insert_to_path(object, path, value) + local curr_obj = object + + for index=1,(#path -1) do + if curr_obj[path[index]] == nil then + curr_obj[path[index]] = {} + end + + curr_obj = curr_obj[path[index]] + end + + curr_obj[path[#path]] = value + end + + local matches = query:iter_matches(qnode, bufnr, start_row, end_row) + + return function() + local pattern, match = matches() + if pattern ~= nil then + local prepared_match = {} + + -- Extract capture names from each match + for id, node in pairs(match) do + local name = query.captures[id] -- name of the capture in the query + if name ~= nil then + local path = split(name) + insert_to_path(prepared_match, path, node) + end + end + + -- Add some predicates for testing + local preds = query.info.patterns[pattern] + if preds then + for _, pred in pairs(preds) do + if pred[1] == "set!" and pred[2] ~= nil then + insert_to_path(prepared_match, split(pred[2]), pred[3]) + end + end + end + + return prepared_match + end + end +end + +return M |
