return { "nvim-treesitter/nvim-treesitter-textobjects", dependencies = { { "nvim-treesitter", build = ":TSUpdate" }, { "nvim-treesitter/nvim-treesitter" }, { "chrisgrieser/nvim-various-textobjs", event = "UIEnter", opts = { keymaps = { useDefaults = true, }, }, }, }, init = function() local wk = require("which-key") wk.add({ { mode = { "n", "v", "x" }, { "g>", group = "Swap next" }, { "g<", group = "Swap prev" }, { "]", group = "Next" }, { "[", group = "Prev" }, }, }) end, config = function() require("nvim-treesitter.configs").setup({ textobjects = { select = { enable = true, -- Automatically jump forward to textobj, similar to targets.vim lookahead = true, keymaps = { -- You can use the capture groups defined in textobjects.scm ["a="] = { query = "@assignment.outer", desc = "Select outer part of an assignment" }, ["i="] = { query = "@assignment.inner", desc = "Select inner part of an assignment" }, ["h="] = { query = "@assignment.lhs", desc = "Select left hand side of an assignment" }, ["l="] = { query = "@assignment.rhs", desc = "Select right hand side of an assignment" }, -- works for javascript/typescript files (custom capture I created in after/queries/ecma/textobjects.scm) ["a:"] = { query = "@property.outer", desc = "Select outer part of an object property" }, ["i:"] = { query = "@property.inner", desc = "Select inner part of an object property" }, ["h:"] = { query = "@property.lhs", desc = "Select left part of an object property" }, ["l:"] = { query = "@property.rhs", desc = "Select right part of an object property" }, ["aa"] = { query = "@parameter.outer", desc = "Select outer part of a parameter/argument" }, ["ia"] = { query = "@parameter.inner", desc = "Select inner part of a parameter/argument" }, ["an"] = { query = "@conditional.outer", desc = "Select outer part of a conditional" }, ["in"] = { query = "@conditional.inner", desc = "Select inner part of a conditional" }, ["ap"] = { query = "@loop.outer", desc = "Select outer part of a loop" }, ["ip"] = { query = "@loop.inner", desc = "Select inner part of a loop" }, ["af"] = { query = "@call.outer", desc = "Select outer part of a function call" }, ["if"] = { query = "@call.inner", desc = "Select inner part of a function call" }, ["am"] = { query = "@function.outer", desc = "Select outer part of a method/function definition", }, ["im"] = { query = "@function.inner", desc = "Select inner part of a method/function definition", }, ["ac"] = { query = "@class.outer", desc = "Select outer part of a class" }, ["ic"] = { query = "@class.inner", desc = "Select inner part of a class" }, }, }, swap = { enable = true, swap_next = { ["g>a"] = { query = "@parameter.inner", desc = "swap parameters/argument with next" }, ["g>:"] = { query = "@property.outer", desc = "swap object property with next" }, ["g>m"] = { query = "@function.outer", desc = "swap function with next" }, }, swap_previous = { ["g/.scm file in your runtime path. -- Below example nvim-treesitter's `locals.scm` and `folds.scm`. They also provide highlights.scm and indent.scm. ["]-"] = { query = "@scope", query_group = "locals", desc = "Next scope" }, ["]z"] = { query = "@fold", query_group = "folds", desc = "Next fold" }, }, goto_next_end = { ["]F"] = { query = "@call.outer", desc = "Next function call end" }, ["]M"] = { query = "@function.outer", desc = "Next method/function def end" }, ["]C"] = { query = "@class.outer", desc = "Next class end" }, ["]+"] = { query = "@conditional.outer", desc = "Next conditional end" }, ["]L"] = { query = "@loop.outer", desc = "Next loop end" }, }, goto_previous_start = { ["[f"] = { query = "@call.outer", desc = "Prev function call start" }, ["[m"] = { query = "@function.outer", desc = "Prev method/function def start" }, ["[c"] = { query = "@class.outer", desc = "Prev class start" }, ["[="] = { query = "@conditional.outer", desc = "Prev conditional start" }, ["[l"] = { query = "@loop.outer", desc = "Prev loop start" }, }, goto_previous_end = { ["[F"] = { query = "@call.outer", desc = "Prev function call end" }, ["[M"] = { query = "@function.outer", desc = "Prev method/function def end" }, ["[C"] = { query = "@class.outer", desc = "Prev class end" }, ["[+"] = { query = "@conditional.outer", desc = "Prev conditional end" }, ["[L"] = { query = "@loop.outer", desc = "Prev loop end" }, }, }, }, }) local ts_repeat_move = require("nvim-treesitter.textobjects.repeatable_move") -- vim way: ; goes to the direction you were moving. vim.keymap.set({ "n", "x", "o" }, ";", ts_repeat_move.repeat_last_move) vim.keymap.set({ "n", "x", "o" }, ",", ts_repeat_move.repeat_last_move_opposite) -- Optionally, make builtin f, F, t, T also repeatable with ; and , vim.keymap.set({ "n", "x", "o" }, "f", ts_repeat_move.builtin_f_expr, { expr = true }) vim.keymap.set({ "n", "x", "o" }, "F", ts_repeat_move.builtin_F_expr, { expr = true }) vim.keymap.set({ "n", "x", "o" }, "t", ts_repeat_move.builtin_t_expr, { expr = true }) vim.keymap.set({ "n", "x", "o" }, "T", ts_repeat_move.builtin_T_expr, { expr = true }) end, }