summaryrefslogtreecommitdiff
path: root/lua/nvim-treesitter/incremental_selection.lua
blob: 9876f2e0175c8ffdf181dd14e19b915ed1270ef2 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
local api = vim.api

local state = require'nvim-treesitter.state'
local configs = require'nvim-treesitter.configs'
local ts_utils = require'nvim-treesitter.ts_utils'

local M = {}

local function update_selection(buf, node)
  local start_row, start_col, end_row, end_col = node:range()

  if end_row == vim.fn.line('$') then
    end_col = #vim.fn.getline('$')
  end

  vim.fn.setpos(".", { buf, start_row+1, start_col+1, 0 })
  vim.fn.nvim_exec("normal v", false)
  vim.fn.setpos(".", { buf, end_row+1, end_col+1, 0 })
end

local function select_incremental(get_parent)
  return function()
    local buf = api.nvim_get_current_buf()
    local buf_state = state.get_buf_state(buf)

    local node
    -- initialize incremental selection with current range
    if #buf_state.selection.nodes == 0 then
      local cur_range = buf_state.selection.range
      if not cur_range then
        local _, cursor_row, cursor_col, _ = unpack(vim.fn.getpos("."))
        cur_range = { cursor_row, cursor_col, cursor_row, cursor_col + 1 }
      end

      local root = buf_state.parser.tree:root()
      if not root then return end

      node = root:named_descendant_for_range(cur_range[1]-1, cur_range[2]-1, cur_range[3]-1, cur_range[4]-1)
    else
      node = get_parent(buf_state.selection.nodes[#buf_state.selection.nodes])
    end

    if not node then return end

    if node ~= buf_state.selection.nodes[#buf_state.selection.nodes] then
      state.insert_selection_node(buf, node)
    end

    update_selection(buf, node)
  end
end

M.node_incremental = select_incremental(function(node)
  return node:parent() or node
end)

M.scope_incremental = select_incremental(function(node)
  return ts_utils.containing_scope(node:parent() or node)
end)

function M.node_decremental()
  local buf = api.nvim_get_current_buf()
  local buf_state = state.get_buf_state(buf)

  local nodes = buf_state.selection.nodes
  if #nodes < 2 then return end

  state.pop_selection_node(buf)

  local node = nodes[#nodes]
  update_selection(buf, node)
end

function M.attach(bufnr)
  local buf = bufnr or api.nvim_get_current_buf()

  local config = configs.get_module('incremental_selection')
  for funcname, mapping in pairs(config.keymaps) do

    if funcname == "init_selection" then
      local cmd = ":lua require'nvim-treesitter.incremental_selection'.node_incremental()<CR>"
      api.nvim_buf_set_keymap(buf, 'n', mapping, cmd, { silent = true })
    else
      local cmd = string.format(":lua require'nvim-treesitter.incremental_selection'.%s()<CR>", funcname)
      api.nvim_buf_set_keymap(buf, 'v', mapping, cmd, { silent = true })
    end
  end
end

function M.detach(bufnr)
  local buf = bufnr or api.nvim_get_current_buf()

  local config = configs.get_module('incremental_selection')
  for f, mapping in pairs(config.keymaps) do
    if f == "init_selection" then
      api.nvim_buf_del_keymap(buf, 'n', mapping)
    else
      api.nvim_buf_del_keymap(buf, 'v', mapping)
    end
  end
end

return M