mirror of
https://github.com/nvim-treesitter/nvim-treesitter.git
synced 2026-07-03 03:56:52 -04:00
fix(fold): revamp fold
fix(fold): typo fix(fold): remove debug and add queries fix(fold): fallback to local scopes for folds
This commit is contained in:
parent
92c18c8efa
commit
545e5c479a
3 changed files with 64 additions and 15 deletions
|
|
@ -1,28 +1,60 @@
|
|||
local api = vim.api
|
||||
local utils = require'nvim-treesitter.ts_utils'
|
||||
local query = require'nvim-treesitter.query'
|
||||
local parsers = require'nvim-treesitter.parsers'
|
||||
|
||||
local M = {}
|
||||
|
||||
local folds_levels = utils.memoize_by_buf_tick(function(bufnr)
|
||||
local lang = parsers.get_buf_lang(bufnr)
|
||||
|
||||
local matches
|
||||
if query.has_fold(lang) then
|
||||
matches = query.get_capture_matches(bufnr, "@fold", "fold")
|
||||
elseif query.has_locals(lang) then
|
||||
matches = query.get_capture_matches(bufnr, "@scope", "locals")
|
||||
else
|
||||
return {}
|
||||
end
|
||||
|
||||
local levels_tmp = {}
|
||||
|
||||
for _, node in ipairs(matches) do
|
||||
local start, _, stop, stop_col = node.node:range()
|
||||
|
||||
if stop_col > 0 then
|
||||
stop = stop + 1
|
||||
end
|
||||
|
||||
-- This can be folded
|
||||
-- Fold only multiline nodes that are not exactly the same as prevsiously met folds
|
||||
if start ~= stop and not (levels_tmp[start] and levels_tmp[stop]) then
|
||||
levels_tmp[start] = (levels_tmp[start] or 0) + 1
|
||||
levels_tmp[stop] = (levels_tmp[stop] or 0) - 1
|
||||
end
|
||||
|
||||
end
|
||||
|
||||
local levels = {}
|
||||
local current_level = 0
|
||||
|
||||
for lnum=0,api.nvim_buf_line_count(bufnr) do
|
||||
current_level = current_level + (levels_tmp[lnum] or 0)
|
||||
levels[lnum + 1] = current_level
|
||||
end
|
||||
|
||||
return levels
|
||||
end)
|
||||
|
||||
function M.get_fold_indic(lnum)
|
||||
if not parsers.has_parser() or not lnum then return '0' end
|
||||
|
||||
local function smallest_multiline_containing(node, level)
|
||||
for index = 0,(node:named_child_count() -1) do
|
||||
local child = node:named_child(index)
|
||||
local start, _, stop, _ = child:range()
|
||||
local buf = api.nvim_get_current_buf()
|
||||
|
||||
if start ~= stop and start <= (lnum -1) and stop >= (lnum -1) then
|
||||
return smallest_multiline_containing(child, level + 1)
|
||||
end
|
||||
end
|
||||
local levels = folds_levels(buf) or {}
|
||||
|
||||
return node, level
|
||||
end
|
||||
return tostring(levels[lnum] or 0)
|
||||
|
||||
local parser = parsers.get_parser()
|
||||
|
||||
local _, level = smallest_multiline_containing(parser:parse():root(), 0)
|
||||
|
||||
return tostring(level)
|
||||
end
|
||||
|
||||
return M
|
||||
|
|
|
|||
8
queries/c/fold.scm
Normal file
8
queries/c/fold.scm
Normal file
|
|
@ -0,0 +1,8 @@
|
|||
[
|
||||
(for_statement)
|
||||
(if_statement)
|
||||
(while_statement)
|
||||
(translation_unit)
|
||||
(function_definition)
|
||||
(compound_statement)
|
||||
] @fold
|
||||
9
queries/lua/fold.scm
Normal file
9
queries/lua/fold.scm
Normal file
|
|
@ -0,0 +1,9 @@
|
|||
[
|
||||
(for_in_statement)
|
||||
(for_statement)
|
||||
(if_statement)
|
||||
(function_definition)
|
||||
(function)
|
||||
(while_statement)
|
||||
(table)
|
||||
] @fold
|
||||
Loading…
Add table
Add a link
Reference in a new issue