aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorThomas Vigouroux <tomvig38@gmail.com>2020-04-23 08:43:35 +0200
committerThomas Vigouroux <tomvig38@gmail.com>2020-04-25 21:46:22 +0200
commitc62685841e52dacb4659f49d16336765e7e10f10 (patch)
tree4a453cb7249de1d910e5cc96fe6d274efab4b08e
parenttextobj: add incremental node selection (diff)
downloadnvim-treesitter-c62685841e52dacb4659f49d16336765e7e10f10.tar
nvim-treesitter-c62685841e52dacb4659f49d16336765e7e10f10.tar.gz
nvim-treesitter-c62685841e52dacb4659f49d16336765e7e10f10.tar.bz2
nvim-treesitter-c62685841e52dacb4659f49d16336765e7e10f10.tar.lz
nvim-treesitter-c62685841e52dacb4659f49d16336765e7e10f10.tar.xz
nvim-treesitter-c62685841e52dacb4659f49d16336765e7e10f10.tar.zst
nvim-treesitter-c62685841e52dacb4659f49d16336765e7e10f10.zip
textobj: add incremental scope selection
-rw-r--r--autoload/nvim_treesitter.vim4
-rw-r--r--lua/nvim-treesitter/textobj.lua22
-rw-r--r--lua/nvim-treesitter/utils.lua15
-rw-r--r--plugin/nvim-treesitter.vim2
4 files changed, 40 insertions, 3 deletions
diff --git a/autoload/nvim_treesitter.vim b/autoload/nvim_treesitter.vim
index 3e5ce40e6..e050ae5b2 100644
--- a/autoload/nvim_treesitter.vim
+++ b/autoload/nvim_treesitter.vim
@@ -11,6 +11,6 @@ function! nvim_treesitter#select_node_incr()
call s:visual_node(luaeval('require"nvim-treesitter.textobj".node_incremental()'))
endfunction
-function! nvim_treesitter#select_context_incr()
- call s:visual_node(luaeval('require"nvim-treesitter.textobj".context_incremental()'))
+function! nvim_treesitter#select_scope_incr()
+ call s:visual_node(luaeval('require"nvim-treesitter.textobj".scope_incremental()'))
endfunction
diff --git a/lua/nvim-treesitter/textobj.lua b/lua/nvim-treesitter/textobj.lua
index 9460e193a..a28440e32 100644
--- a/lua/nvim-treesitter/textobj.lua
+++ b/lua/nvim-treesitter/textobj.lua
@@ -33,4 +33,26 @@ function M.node_incremental()
end
end
+function M.scope_incremental()
+ local _, sel_start_line, sel_start_col, _ = unpack(vim.fn.getpos("'<"))
+ local _, sel_end_line, sel_end_col, _ = unpack(vim.fn.getpos("'>"))
+
+ if parsers.has_parser() then
+ local root = parsers.get_parser():parse():root()
+ local node = utils.smallest_containing_scope(
+ root:named_descendant_for_range(sel_start_line-1, sel_start_col-1, sel_end_line-1, sel_end_col))
+
+ local node_start_row, node_start_col, node_end_row, node_end_col = node:range()
+
+ if (sel_start_line-1) == node_start_row and (sel_start_col-1) == node_start_col
+ and (sel_end_line-1) == node_end_row and sel_end_col == node_end_col then
+ return node_range_to_vim(utils.smallest_containing_scope(node:parent() or node))
+ else
+ return node_range_to_vim(node)
+ end
+ else
+ return node_range_to_vim()
+ end
+end
+
return M
diff --git a/lua/nvim-treesitter/utils.lua b/lua/nvim-treesitter/utils.lua
index 59de5dd57..2ef2746aa 100644
--- a/lua/nvim-treesitter/utils.lua
+++ b/lua/nvim-treesitter/utils.lua
@@ -1,6 +1,7 @@
-- Utils collection for nvim-treesitter
local api = vim.api
local parsers = require'nvim-treesitter.parsers'
+local locals = require'nvim-treesitter.locals'
local M = {}
@@ -74,6 +75,20 @@ function M.setup_commands(mod, commands)
})
api.nvim_command(table.concat(parts, " "))
end
+--- Gets the smallest scope which contains @param node
+function M.smallest_containing_scope(node, bufnr)
+ local bufnr = bufnr or api.nvim_get_current_buf()
+
+ local root = parsers.get_parser(bufnr):parse():root()
+ if not node then return root end
+
+ local scopes = locals.get_scopes(bufnr)
+ local current = node
+ while current ~= nil and not vim.tbl_contains(scopes, current) do
+ current = current:parent()
+ end
+
+ return current or root
end
return M
diff --git a/plugin/nvim-treesitter.vim b/plugin/nvim-treesitter.vim
index a94f56a96..31329c61e 100644
--- a/plugin/nvim-treesitter.vim
+++ b/plugin/nvim-treesitter.vim
@@ -1,4 +1,4 @@
-" Last Change: 2020 avril 19
+" Last Change: 2020 avril 23
if exists('g:loaded_nvim_treesitter')
finish