aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorDylan Kendal <dylankendal@gmail.com>2021-08-20 01:35:25 -0400
committerDylan Kendal <dylankendal@gmail.com>2021-08-20 11:58:15 -0400
commit140084180e82bea3afafa32a33fee37e6e9473ca (patch)
tree73c33ba7ddb3fd954fa78f6003c55997b836266f
parent2ae9ff128583984145ce38e61bbe9047871616bf (diff)
downloadrneovim-140084180e82bea3afafa32a33fee37e6e9473ca.tar.gz
rneovim-140084180e82bea3afafa32a33fee37e6e9473ca.tar.bz2
rneovim-140084180e82bea3afafa32a33fee37e6e9473ca.zip
feat(treesitter): add next, prev sibling method
Add tsnode methods to change to the next, previous, named or unnamed nodes.
-rw-r--r--runtime/doc/treesitter.txt12
-rw-r--r--src/nvim/lua/treesitter.c48
-rw-r--r--test/functional/treesitter/node_spec.lua62
3 files changed, 122 insertions, 0 deletions
diff --git a/runtime/doc/treesitter.txt b/runtime/doc/treesitter.txt
index 906e2d716a..379523722e 100644
--- a/runtime/doc/treesitter.txt
+++ b/runtime/doc/treesitter.txt
@@ -86,6 +86,18 @@ Node methods *lua-treesitter-node*
tsnode:parent() *tsnode:parent()*
Get the node's immediate parent.
+tsnode:next_sibling() *tsnode:next_sibling()*
+ Get the node's next sibling.
+
+tsnode:prev_sibling() *tsnode:prev_sibling()*
+ Get the node's previous sibling.
+
+tsnode:next_named_sibling() *tsnode:next_named_sibling()*
+ Get the node's next named sibling.
+
+tsnode:prev_named_sibling() *tsnode:prev_named_sibling()*
+ Get the node's previous named sibling.
+
tsnode:iter_children() *tsnode:iter_children()*
Iterates over all the direct children of {tsnode}, regardless of
wether they are named or not.
diff --git a/src/nvim/lua/treesitter.c b/src/nvim/lua/treesitter.c
index 1425baacf0..2dfcd9a958 100644
--- a/src/nvim/lua/treesitter.c
+++ b/src/nvim/lua/treesitter.c
@@ -80,6 +80,10 @@ static struct luaL_Reg node_meta[] = {
{ "parent", node_parent },
{ "iter_children", node_iter_children },
{ "_rawquery", node_rawquery },
+ { "next_sibling", node_next_sibling },
+ { "prev_sibling", node_prev_sibling },
+ { "next_named_sibling", node_next_named_sibling },
+ { "prev_named_sibling", node_prev_named_sibling },
{ NULL, NULL }
};
@@ -992,6 +996,50 @@ static int node_parent(lua_State *L)
return 1;
}
+static int node_next_sibling(lua_State *L)
+{
+ TSNode node;
+ if (!node_check(L, 1, &node)) {
+ return 0;
+ }
+ TSNode sibling = ts_node_next_sibling(node);
+ push_node(L, sibling, 1);
+ return 1;
+}
+
+static int node_prev_sibling(lua_State *L)
+{
+ TSNode node;
+ if (!node_check(L, 1, &node)) {
+ return 0;
+ }
+ TSNode sibling = ts_node_prev_sibling(node);
+ push_node(L, sibling, 1);
+ return 1;
+}
+
+static int node_next_named_sibling(lua_State *L)
+{
+ TSNode node;
+ if (!node_check(L, 1, &node)) {
+ return 0;
+ }
+ TSNode sibling = ts_node_next_named_sibling(node);
+ push_node(L, sibling, 1);
+ return 1;
+}
+
+static int node_prev_named_sibling(lua_State *L)
+{
+ TSNode node;
+ if (!node_check(L, 1, &node)) {
+ return 0;
+ }
+ TSNode sibling = ts_node_prev_named_sibling(node);
+ push_node(L, sibling, 1);
+ return 1;
+}
+
/// assumes the match table being on top of the stack
static void set_match(lua_State *L, TSQueryMatch *match, int nodeidx)
{
diff --git a/test/functional/treesitter/node_spec.lua b/test/functional/treesitter/node_spec.lua
new file mode 100644
index 0000000000..21c287644e
--- /dev/null
+++ b/test/functional/treesitter/node_spec.lua
@@ -0,0 +1,62 @@
+local helpers = require('test.functional.helpers')(after_each)
+
+local clear = helpers.clear
+local eq = helpers.eq
+local exec_lua = helpers.exec_lua
+local insert = helpers.insert
+local pending_c_parser = helpers.pending_c_parser
+
+before_each(clear)
+
+local function lua_eval(lua_expr)
+ return exec_lua("return " .. lua_expr)
+end
+
+describe('treesitter node API', function()
+ clear()
+
+ if pending_c_parser(pending) then
+ return
+ end
+
+ it('can move between siblings', function()
+ insert([[
+ int main(int x, int y, int z) {
+ return x + y * z
+ }
+ ]])
+
+ exec_lua([[
+ query = require"vim.treesitter.query"
+ parser = vim.treesitter.get_parser(0, "c")
+ tree = parser:parse()[1]
+ root = tree:root()
+ lang = vim.treesitter.inspect_language('c')
+
+ function node_text(node)
+ return query.get_node_text(node, 0)
+ end
+ ]])
+
+ exec_lua 'node = root:descendant_for_range(0, 11, 0, 16)'
+ eq('int x', lua_eval('node_text(node)'))
+
+ exec_lua 'node = node:next_sibling()'
+ eq(',', lua_eval('node_text(node)'))
+
+ exec_lua 'node = node:next_sibling()'
+ eq('int y', lua_eval('node_text(node)'))
+
+ exec_lua 'node = node:prev_sibling()'
+ eq(',', lua_eval('node_text(node)'))
+
+ exec_lua 'node = node:prev_sibling()'
+ eq('int x', lua_eval('node_text(node)'))
+
+ exec_lua 'node = node:next_named_sibling()'
+ eq('int y', lua_eval('node_text(node)'))
+
+ exec_lua 'node = node:prev_named_sibling()'
+ eq('int x', lua_eval('node_text(node)'))
+ end)
+end)